Fix RP-Version 8.0 Compatibility, #57

This commit is contained in:
Florian Märkl 2019-10-11 15:24:46 +02:00
commit cf6829287a
No known key found for this signature in database
GPG key ID: 125BC8A5A6A1E857
4 changed files with 128 additions and 38 deletions

View file

@ -42,12 +42,23 @@ extern "C" {
#define CHIAKI_RP_APPLICATION_REASON_INVALID_PSN_ID 0x80108b02 #define CHIAKI_RP_APPLICATION_REASON_INVALID_PSN_ID 0x80108b02
#define CHIAKI_RP_APPLICATION_REASON_IN_USE 0x80108b10 #define CHIAKI_RP_APPLICATION_REASON_IN_USE 0x80108b10
#define CHIAKI_RP_APPLICATION_REASON_CRASH 0x80108b15 #define CHIAKI_RP_APPLICATION_REASON_CRASH 0x80108b15
#define CHIAKI_RP_APPLICATION_REASON_CLIENT_OUTDATED 0x80108b11 #define CHIAKI_RP_APPLICATION_REASON_RP_VERSION 0x80108b11
// unknown: 0x80108bff // unknown: 0x80108bff
#define CHIAKI_RP_CLIENT_VERSION "9.0" CHIAKI_EXPORT const char *chiaki_rp_application_reason_string(uint32_t reason);
const char *chiaki_rp_application_reason_string(uint32_t reason); typedef enum {
CHIAKI_RP_VERSION_UNKNOWN = 0,
CHIAKI_RP_VERSION_8_0 = 800,
CHIAKI_RP_VERSION_9_0 = 900
} ChiakiRpVersion;
/**
* @return RP-Version string or NULL
*/
CHIAKI_EXPORT const char *chiaki_rp_version_string(ChiakiRpVersion version);
CHIAKI_EXPORT ChiakiRpVersion chiaki_rp_version_parse(const char *rp_version_str);
#define CHIAKI_RP_DID_SIZE 32 #define CHIAKI_RP_DID_SIZE 32
@ -94,6 +105,7 @@ typedef enum {
CHIAKI_QUIT_REASON_SESSION_REQUEST_CONNECTION_REFUSED, CHIAKI_QUIT_REASON_SESSION_REQUEST_CONNECTION_REFUSED,
CHIAKI_QUIT_REASON_SESSION_REQUEST_RP_IN_USE, CHIAKI_QUIT_REASON_SESSION_REQUEST_RP_IN_USE,
CHIAKI_QUIT_REASON_SESSION_REQUEST_RP_CRASH, CHIAKI_QUIT_REASON_SESSION_REQUEST_RP_CRASH,
CHIAKI_QUIT_REASON_SESSION_REQUEST_RP_VERSION_MISMATCH,
CHIAKI_QUIT_REASON_CTRL_UNKNOWN, CHIAKI_QUIT_REASON_CTRL_UNKNOWN,
CHIAKI_QUIT_REASON_CTRL_CONNECT_FAILED, CHIAKI_QUIT_REASON_CTRL_CONNECT_FAILED,
CHIAKI_QUIT_REASON_CTRL_CONNECTION_REFUSED, CHIAKI_QUIT_REASON_CTRL_CONNECTION_REFUSED,
@ -156,6 +168,8 @@ typedef struct chiaki_session_t
ChiakiConnectVideoProfile video_profile; ChiakiConnectVideoProfile video_profile;
} connect_info; } connect_info;
ChiakiRpVersion rp_version;
uint8_t nonce[CHIAKI_RPCRYPT_KEY_SIZE]; uint8_t nonce[CHIAKI_RPCRYPT_KEY_SIZE];
ChiakiRPCrypt rpcrypt; ChiakiRPCrypt rpcrypt;
char session_id[CHIAKI_SESSION_ID_SIZE_MAX]; // zero-terminated char session_id[CHIAKI_SESSION_ID_SIZE_MAX]; // zero-terminated

View file

@ -570,16 +570,18 @@ static ChiakiErrorCode ctrl_connect(ChiakiCtrl *ctrl)
"Connection: keep-alive\r\n" "Connection: keep-alive\r\n"
"Content-Length: 0\r\n" "Content-Length: 0\r\n"
"RP-Auth: %s\r\n" "RP-Auth: %s\r\n"
"RP-Version: " CHIAKI_RP_CLIENT_VERSION "\r\n" "RP-Version: %s\r\n"
"RP-Did: %s\r\n" "RP-Did: %s\r\n"
"RP-ControllerType: 3\r\n" "RP-ControllerType: 3\r\n"
"RP-ClientType: 11\r\n" "RP-ClientType: 11\r\n"
"RP-OSType: %s\r\n" "RP-OSType: %s\r\n"
"RP-ConPath: 1\r\n\r\n"; "RP-ConPath: 1\r\n\r\n";
const char *rp_version = chiaki_rp_version_string(session->rp_version);
char buf[512]; char buf[512];
int request_len = snprintf(buf, sizeof(buf), request_fmt, int request_len = snprintf(buf, sizeof(buf), request_fmt,
session->connect_info.hostname, SESSION_CTRL_PORT, auth_b64, did_b64, ostype_b64); session->connect_info.hostname, SESSION_CTRL_PORT, auth_b64, rp_version ? rp_version : "", did_b64, ostype_b64);
if(request_len < 0 || request_len >= sizeof(buf)) if(request_len < 0 || request_len >= sizeof(buf))
goto error; goto error;

View file

@ -106,13 +106,16 @@ static void regist_event_simple(ChiakiRegist *regist, ChiakiRegistEventType type
regist->cb(&event, regist->cb_user); regist->cb(&event, regist->cb_user);
} }
static const char * const request_fmt = static const char * const request_head_fmt =
"POST /sce/rp/regist HTTP/1.1\r\n" "POST /sce/rp/regist HTTP/1.1\r\n"
"HOST: 10.0.2.15\r\n" // random lol "HOST: 10.0.2.15\r\n" // random lol
"User-Agent: remoteplay Windows\r\n" "User-Agent: remoteplay Windows\r\n"
"Connection: close\r\n" "Connection: close\r\n"
"Content-Length: %llu\r\n" "Content-Length: %llu\r\n";
"RP-Version: " CHIAKI_RP_CLIENT_VERSION "\r\n\r\n";
static const char * const request_rp_version_fmt = "RP-Version: %s\r\n";
static const char * const request_tail = "\r\n";
static const char * const request_inner_account_id_fmt = static const char * const request_inner_account_id_fmt =
"Client-Type: Windows\r\n" "Client-Type: Windows\r\n"
@ -123,6 +126,28 @@ static const char * const request_inner_online_id_fmt =
"Np-Online-Id: %s\r\n"; "Np-Online-Id: %s\r\n";
static int request_header_format(char *buf, size_t buf_size, size_t payload_size, ChiakiRpVersion rp_version)
{
int cur = snprintf(buf, buf_size, request_head_fmt, (unsigned long long)payload_size);
if(cur < 0 || cur >= payload_size)
return -1;
if(rp_version >= CHIAKI_RP_VERSION_9_0)
{
const char *rp_version_str = chiaki_rp_version_string(rp_version);
size_t s = buf_size - cur;
int r = snprintf(buf + cur, s, request_rp_version_fmt, rp_version_str);
if(r < 0 || r >= s)
return -1;
cur += r;
}
size_t tail_size = strlen(request_tail) + 1;
if(cur + tail_size > payload_size)
return -1;
memcpy(buf + cur, request_tail, tail_size);
cur += (int)tail_size - 1;
return cur;
}
CHIAKI_EXPORT ChiakiErrorCode chiaki_regist_request_payload_format(uint8_t *buf, size_t *buf_size, ChiakiRPCrypt *crypt, const char *psn_online_id, const uint8_t *psn_account_id) CHIAKI_EXPORT ChiakiErrorCode chiaki_regist_request_payload_format(uint8_t *buf, size_t *buf_size, ChiakiRPCrypt *crypt, const char *psn_online_id, const uint8_t *psn_account_id)
{ {
@ -179,14 +204,19 @@ static void *regist_thread_func(void *user)
goto fail; goto fail;
} }
ChiakiRpVersion rp_version = regist->info.psn_online_id ? CHIAKI_RP_VERSION_8_0 : CHIAKI_RP_VERSION_9_0;
char request_header[0x100]; char request_header[0x100];
int request_header_size = snprintf(request_header, sizeof(request_header), request_fmt, (unsigned long long)payload_size); int request_header_size = request_header_format(request_header, sizeof(request_header), payload_size, rp_version);
if(request_header_size >= sizeof(request_header))
if(request_header_size < 0 || request_header_size >= sizeof(request_header))
{ {
CHIAKI_LOGE(regist->log, "Regist failed to format request"); CHIAKI_LOGE(regist->log, "Regist failed to format request");
goto fail; goto fail;
} }
CHIAKI_LOGV(regist->log, "Regist formatted request header:");
chiaki_log_hexdump(regist->log, CHIAKI_LOG_VERBOSE, (uint8_t *)request_header, request_header_size);
struct addrinfo *addrinfos; struct addrinfo *addrinfos;
int r = getaddrinfo(regist->info.host, NULL, NULL, &addrinfos); int r = getaddrinfo(regist->info.host, NULL, NULL, &addrinfos);
if(r != 0) if(r != 0)

View file

@ -44,6 +44,8 @@
#define SESSION_EXPECT_TIMEOUT_MS 5000 #define SESSION_EXPECT_TIMEOUT_MS 5000
static void *session_thread_func(void *arg);
static bool session_thread_request_session(ChiakiSession *session, ChiakiRpVersion *server_version_out);
const char *chiaki_rp_application_reason_string(uint32_t reason) const char *chiaki_rp_application_reason_string(uint32_t reason)
{ {
@ -57,13 +59,34 @@ const char *chiaki_rp_application_reason_string(uint32_t reason)
return "Remote is already in use"; return "Remote is already in use";
case CHIAKI_RP_APPLICATION_REASON_CRASH: case CHIAKI_RP_APPLICATION_REASON_CRASH:
return "Remote Play on Console crashed"; return "Remote Play on Console crashed";
case CHIAKI_RP_APPLICATION_REASON_CLIENT_OUTDATED: case CHIAKI_RP_APPLICATION_REASON_RP_VERSION:
return "Client outdated"; return "RP-Version mismatch";
default: default:
return "unknown"; return "unknown";
} }
} }
const char *chiaki_rp_version_string(ChiakiRpVersion version)
{
switch(version)
{
case CHIAKI_RP_VERSION_8_0:
return "9.0";
case CHIAKI_RP_VERSION_9_0:
return "9.0";
default:
return NULL;
}
}
CHIAKI_EXPORT ChiakiRpVersion chiaki_rp_version_parse(const char *rp_version_str)
{
if(strcmp(rp_version_str, "8.0") == 0)
return CHIAKI_RP_VERSION_8_0;
if(strcmp(rp_version_str, "9.0") == 0)
return CHIAKI_RP_VERSION_9_0;
return CHIAKI_RP_VERSION_UNKNOWN;
}
CHIAKI_EXPORT void chiaki_connect_video_profile_preset(ChiakiConnectVideoProfile *profile, ChiakiVideoResolutionPreset resolution, ChiakiVideoFPSPreset fps) CHIAKI_EXPORT void chiaki_connect_video_profile_preset(ChiakiConnectVideoProfile *profile, ChiakiVideoResolutionPreset resolution, ChiakiVideoFPSPreset fps)
{ {
@ -124,6 +147,8 @@ CHIAKI_EXPORT const char *chiaki_quit_reason_string(ChiakiQuitReason reason)
return "Remote Play on Console is already in use"; return "Remote Play on Console is already in use";
case CHIAKI_QUIT_REASON_SESSION_REQUEST_RP_CRASH: case CHIAKI_QUIT_REASON_SESSION_REQUEST_RP_CRASH:
return "Remote Play on Console has crashed"; return "Remote Play on Console has crashed";
case CHIAKI_QUIT_REASON_SESSION_REQUEST_RP_VERSION_MISMATCH:
return "RP-Version mismatch";
case CHIAKI_QUIT_REASON_CTRL_UNKNOWN: case CHIAKI_QUIT_REASON_CTRL_UNKNOWN:
return "Unknown Ctrl Error"; return "Unknown Ctrl Error";
case CHIAKI_QUIT_REASON_CTRL_CONNECTION_REFUSED: case CHIAKI_QUIT_REASON_CTRL_CONNECTION_REFUSED:
@ -140,7 +165,6 @@ CHIAKI_EXPORT const char *chiaki_quit_reason_string(ChiakiQuitReason reason)
} }
} }
static void *session_thread_func(void *arg);
CHIAKI_EXPORT ChiakiErrorCode chiaki_session_init(ChiakiSession *session, ChiakiConnectInfo *connect_info, ChiakiLog *log) CHIAKI_EXPORT ChiakiErrorCode chiaki_session_init(ChiakiSession *session, ChiakiConnectInfo *connect_info, ChiakiLog *log)
@ -148,8 +172,8 @@ CHIAKI_EXPORT ChiakiErrorCode chiaki_session_init(ChiakiSession *session, Chiaki
memset(session, 0, sizeof(ChiakiSession)); memset(session, 0, sizeof(ChiakiSession));
session->log = log; session->log = log;
session->quit_reason = CHIAKI_QUIT_REASON_NONE; session->quit_reason = CHIAKI_QUIT_REASON_NONE;
session->rp_version = CHIAKI_RP_VERSION_9_0;
ChiakiErrorCode err = chiaki_cond_init(&session->state_cond); ChiakiErrorCode err = chiaki_cond_init(&session->state_cond);
if(err != CHIAKI_ERR_SUCCESS) if(err != CHIAKI_ERR_SUCCESS)
@ -288,8 +312,6 @@ void chiaki_session_send_event(ChiakiSession *session, ChiakiEvent *event)
} }
static bool session_thread_request_session(ChiakiSession *session);
static bool session_check_state_pred(void *user) static bool session_check_state_pred(void *user)
{ {
ChiakiSession *session = user; ChiakiSession *session = user;
@ -346,7 +368,16 @@ static void *session_thread_func(void *arg)
CHIAKI_LOGI(session->log, "Starting session request"); CHIAKI_LOGI(session->log, "Starting session request");
success = session_thread_request_session(session); ChiakiRpVersion server_rp_version = CHIAKI_RP_VERSION_UNKNOWN;
success = session_thread_request_session(session, &server_rp_version);
if(!success && server_rp_version != CHIAKI_RP_VERSION_UNKNOWN)
{
CHIAKI_LOGI(session->log, "Attempting to re-request session with Server's RP-Version");
session->rp_version = server_rp_version;
success = session_thread_request_session(session, NULL);
}
if(!success) if(!success)
QUIT(quit); QUIT(quit);
@ -536,32 +567,27 @@ static void parse_session_response(SessionResponse *response, ChiakiHttpResponse
{ {
memset(response, 0, sizeof(SessionResponse)); memset(response, 0, sizeof(SessionResponse));
if(http_response->code == 200)
{
for(ChiakiHttpHeader *header=http_response->headers; header; header=header->next) for(ChiakiHttpHeader *header=http_response->headers; header; header=header->next)
{ {
if(strcmp(header->key, "RP-Nonce") == 0) if(strcmp(header->key, "RP-Nonce") == 0)
response->nonce = header->value; response->nonce = header->value;
else if(strcmp(header->key, "RP-Version") == 0) else if(strcasecmp(header->key, "RP-Version") == 0)
response->rp_version = header->value; response->rp_version = header->value;
} else if(strcmp(header->key, "RP-Application-Reason") == 0)
response->success = response->nonce != NULL;
}
else
{
for(ChiakiHttpHeader *header=http_response->headers; header; header=header->next)
{
if(strcmp(header->key, "RP-Application-Reason") == 0)
{
response->error_code = (uint32_t)strtoul(header->value, NULL, 0x10); response->error_code = (uint32_t)strtoul(header->value, NULL, 0x10);
} }
}
if(http_response->code == 200)
response->success = response->nonce != NULL;
else
response->success = false; response->success = false;
}
} }
static bool session_thread_request_session(ChiakiSession *session) /**
* @param server_version_out if NULL, version mismatch means to fail the entire session, otherwise report the version here
*/
static bool session_thread_request_session(ChiakiSession *session, ChiakiRpVersion *server_version_out)
{ {
chiaki_socket_t session_sock = CHIAKI_INVALID_SOCKET; chiaki_socket_t session_sock = CHIAKI_INVALID_SOCKET;
for(struct addrinfo *ai=session->connect_info.host_addrinfos; ai; ai=ai->ai_next) for(struct addrinfo *ai=session->connect_info.host_addrinfos; ai; ai=ai->ai_next)
@ -650,7 +676,7 @@ static bool session_thread_request_session(ChiakiSession *session)
"Connection: close\r\n" "Connection: close\r\n"
"Content-Length: 0\r\n" "Content-Length: 0\r\n"
"RP-Registkey: %s\r\n" "RP-Registkey: %s\r\n"
"Rp-Version: " CHIAKI_RP_CLIENT_VERSION "\r\n" "Rp-Version: %s\r\n"
"\r\n"; "\r\n";
size_t regist_key_len = sizeof(session->connect_info.regist_key); size_t regist_key_len = sizeof(session->connect_info.regist_key);
@ -671,9 +697,11 @@ static bool session_thread_request_session(ChiakiSession *session)
return false; return false;
} }
const char *rp_version_str = chiaki_rp_version_string(session->rp_version);
char buf[512]; char buf[512];
int request_len = snprintf(buf, sizeof(buf), session_request_fmt, int request_len = snprintf(buf, sizeof(buf), session_request_fmt,
session->connect_info.hostname, SESSION_PORT, regist_key_hex); session->connect_info.hostname, SESSION_PORT, regist_key_hex, rp_version_str ? rp_version_str : "");
if(request_len < 0 || request_len >= sizeof(buf)) if(request_len < 0 || request_len >= sizeof(buf))
{ {
CHIAKI_SOCKET_CLOSE(session_sock); CHIAKI_SOCKET_CLOSE(session_sock);
@ -682,6 +710,7 @@ static bool session_thread_request_session(ChiakiSession *session)
} }
CHIAKI_LOGI(session->log, "Sending session request"); CHIAKI_LOGI(session->log, "Sending session request");
chiaki_log_hexdump(session->log, CHIAKI_LOG_VERBOSE, (uint8_t *)buf, request_len);
int sent = send(session_sock, buf, (size_t)request_len, 0); int sent = send(session_sock, buf, (size_t)request_len, 0);
if(sent < 0) if(sent < 0)
@ -739,6 +768,18 @@ static bool session_thread_request_session(ChiakiSession *session)
session->quit_reason = CHIAKI_QUIT_REASON_SESSION_REQUEST_UNKNOWN; session->quit_reason = CHIAKI_QUIT_REASON_SESSION_REQUEST_UNKNOWN;
} }
} }
else if(response.error_code == CHIAKI_RP_APPLICATION_REASON_RP_VERSION && server_version_out && response.rp_version)
{
CHIAKI_LOGI(session->log, "Reported RP-Version mismatch. ours = %s, server = %s", rp_version_str ? rp_version_str : "", response.rp_version);
*server_version_out = chiaki_rp_version_parse(response.rp_version);
if(*server_version_out != CHIAKI_RP_VERSION_UNKNOWN)
CHIAKI_LOGI(session->log, "Detected Server RP-Version %s", chiaki_rp_version_string(*server_version_out));
else
{
CHIAKI_LOGE(session->log, "Server RP-Version is unknown");
session->quit_reason = CHIAKI_QUIT_REASON_SESSION_REQUEST_RP_VERSION_MISMATCH;
}
}
else else
{ {
CHIAKI_LOGE(session->log, "Reported Application Reason: %#x (%s)", (unsigned int)response.error_code, chiaki_rp_application_reason_string(response.error_code)); CHIAKI_LOGE(session->log, "Reported Application Reason: %#x (%s)", (unsigned int)response.error_code, chiaki_rp_application_reason_string(response.error_code));
@ -750,6 +791,9 @@ static bool session_thread_request_session(ChiakiSession *session)
case CHIAKI_RP_APPLICATION_REASON_CRASH: case CHIAKI_RP_APPLICATION_REASON_CRASH:
session->quit_reason = CHIAKI_QUIT_REASON_SESSION_REQUEST_RP_CRASH; session->quit_reason = CHIAKI_QUIT_REASON_SESSION_REQUEST_RP_CRASH;
break; break;
case CHIAKI_RP_APPLICATION_REASON_RP_VERSION:
session->quit_reason = CHIAKI_QUIT_REASON_SESSION_REQUEST_RP_VERSION_MISMATCH;
break;
default: default:
session->quit_reason = CHIAKI_QUIT_REASON_SESSION_REQUEST_UNKNOWN; session->quit_reason = CHIAKI_QUIT_REASON_SESSION_REQUEST_UNKNOWN;
break; break;