/** * FreeRDP: A Remote Desktop Protocol Client * RDP Licensing * * Copyright 2011 Marc-Andre Moreau * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "license.h" uint8 LICENSE_MESSAGE_STRINGS[][32] = { "", "License Request", "Platform Challenge", "New License", "Upgrade License", "", "", "", "", "", "", "", "", "", "", "", "", "", "License Info", "New License Request", "", "Platform Challenge Response", "", "", "", "", "", "", "", "", "", "Error Alert" }; uint8 error_codes[][32] = { "ERR_UNKNOWN", "ERR_INVALID_SERVER_CERTIFICATE", "ERR_NO_LICENSE", "ERR_INVALID_MAC", "ERR_INVALID_SCOPE", "ERR_UNKNOWN", "ERR_NO_LICENSE_SERVER", "STATUS_VALID_CLIENT", "ERR_INVALID_CLIENT", "ERR_UNKNOWN", "ERR_UNKNOWN", "ERR_INVALID_PRODUCT_ID", "ERR_INVALID_MESSAGE_LENGTH" }; uint8 state_transitions[][32] = { "ST_UNKNOWN", "ST_TOTAL_ABORT", "ST_NO_TRANSITION", "ST_RESET_PHASE_TO_START", "ST_RESEND_LAST_MESSAGE" }; /** * Read a licensing preamble.\n * @msdn{cc240480} * @param s stream * @param bMsgType license message type * @param flags message flags * @param wMsgSize message size */ void license_read_preamble(STREAM* s, uint8* bMsgType, uint8* flags, uint16* wMsgSize) { /* preamble (4 bytes) */ stream_read_uint8(s, *bMsgType); /* bMsgType (1 byte) */ stream_read_uint8(s, *flags); /* flags (1 byte) */ stream_read_uint16(s, *wMsgSize); /* wMsgSize (2 bytes) */ } /** * Write a licensing preamble.\n * @msdn{cc240480} * @param s stream * @param bMsgType license message type * @param flags message flags * @param wMsgSize message size */ void license_write_preamble(STREAM* s, uint8 bMsgType, uint8 flags, uint16 wMsgSize) { /* preamble (4 bytes) */ stream_write_uint8(s, bMsgType); /* bMsgType (1 byte) */ stream_write_uint8(s, flags); /* flags (1 byte) */ stream_write_uint16(s, wMsgSize); /* wMsgSize (2 bytes) */ } /** * Initialize a license packet stream.\n * @param license license module * @return stream */ STREAM* license_send_stream_init(rdpLicense* license) { STREAM* s; s = transport_send_stream_init(license->rdp->transport, 4096); stream_seek(s, LICENSE_PACKET_HEADER_LENGTH); return s; } /** * Send an RDP licensing packet.\n * @msdn{cc240479} * @param license license module * @param s stream */ boolean license_send(rdpLicense* license, STREAM* s, uint8 type) { int length; uint8 flags; uint16 wMsgSize; uint16 sec_flags; DEBUG_LICENSE("Sending %s Packet", LICENSE_MESSAGE_STRINGS[type & 0x1F]); length = stream_get_length(s); stream_set_pos(s, 0); sec_flags = SEC_LICENSE_PKT; wMsgSize = length - LICENSE_PACKET_HEADER_LENGTH + 4; /** * Using EXTENDED_ERROR_MSG_SUPPORTED here would cause mstsc to crash when * running in server mode! This flag seems to be incorrectly documented. */ flags = PREAMBLE_VERSION_3_0; rdp_write_header(license->rdp, s, length, MCS_GLOBAL_CHANNEL_ID); rdp_write_security_header(s, sec_flags); license_write_preamble(s, type, flags, wMsgSize); #ifdef WITH_DEBUG_LICENSE printf("Sending %s Packet, length %d\n", LICENSE_MESSAGE_STRINGS[type & 0x1F], wMsgSize); freerdp_hexdump(s->p - 4, wMsgSize); #endif stream_set_pos(s, length); if (transport_write(license->rdp->transport, s) < 0) return False; return True; } /** * Receive an RDP licensing packet.\n * @msdn{cc240479} * @param license license module * @param s stream */ boolean license_recv(rdpLicense* license, STREAM* s) { uint16 length; uint16 channelId; uint16 sec_flags; uint8 flags; uint8 bMsgType; uint16 wMsgSize; if (!rdp_read_header(license->rdp, s, &length, &channelId)) { printf("Incorrect RDP header.\n"); return False; } rdp_read_security_header(s, &sec_flags); if (!(sec_flags & SEC_LICENSE_PKT)) { printf("Unexpected license packet.\n"); return False; } license_read_preamble(s, &bMsgType, &flags, &wMsgSize); /* preamble (4 bytes) */ DEBUG_LICENSE("Receiving %s Packet", LICENSE_MESSAGE_STRINGS[bMsgType & 0x1F]); switch (bMsgType) { case LICENSE_REQUEST: license_read_license_request_packet(license, s); license_send_new_license_request_packet(license); break; case PLATFORM_CHALLENGE: license_read_platform_challenge_packet(license, s); license_send_platform_challenge_response_packet(license); break; case NEW_LICENSE: license_read_new_license_packet(license, s); break; case UPGRADE_LICENSE: license_read_upgrade_license_packet(license, s); break; case ERROR_ALERT: license_read_error_alert_packet(license, s); break; default: printf("invalid bMsgType:%d\n", bMsgType); return False; } return True; } void license_generate_randoms(rdpLicense* license) { #if 0 crypto_nonce(license->client_random, CLIENT_RANDOM_LENGTH); /* ClientRandom */ crypto_nonce(license->premaster_secret, PREMASTER_SECRET_LENGTH); /* PremasterSecret */ #else memset(license->client_random, 0, CLIENT_RANDOM_LENGTH); /* ClientRandom */ memset(license->premaster_secret, 0, PREMASTER_SECRET_LENGTH); /* PremasterSecret */ #endif } /** * Generate License Cryptographic Keys. * @param license license module */ void license_generate_keys(rdpLicense* license) { security_master_secret(license->premaster_secret, license->client_random, license->server_random, license->master_secret); /* MasterSecret */ security_session_key_blob(license->master_secret, license->client_random, license->server_random, license->session_key_blob); /* SessionKeyBlob */ security_mac_salt_key(license->session_key_blob, license->client_random, license->server_random, license->mac_salt_key); /* MacSaltKey */ security_licensing_encryption_key(license->session_key_blob, license->client_random, license->server_random, license->licensing_encryption_key); /* LicensingEncryptionKey */ #ifdef WITH_DEBUG_LICENSE printf("ClientRandom:\n"); freerdp_hexdump(license->client_random, CLIENT_RANDOM_LENGTH); printf("ServerRandom:\n"); freerdp_hexdump(license->server_random, SERVER_RANDOM_LENGTH); printf("PremasterSecret:\n"); freerdp_hexdump(license->premaster_secret, PREMASTER_SECRET_LENGTH); printf("MasterSecret:\n"); freerdp_hexdump(license->master_secret, MASTER_SECRET_LENGTH); printf("SessionKeyBlob:\n"); freerdp_hexdump(license->session_key_blob, SESSION_KEY_rdpBlob_LENGTH); printf("MacSaltKey:\n"); freerdp_hexdump(license->mac_salt_key, MAC_SALT_KEY_LENGTH); printf("LicensingEncryptionKey:\n"); freerdp_hexdump(license->licensing_encryption_key, LICENSING_ENCRYPTION_KEY_LENGTH); #endif } /** * Generate Unique Hardware Identifier (CLIENT_HARDWARE_ID).\n * @param license license module */ void license_generate_hwid(rdpLicense* license) { CryptoMd5 md5; uint8* mac_address; memset(license->hwid, 0, HWID_LENGTH); mac_address = license->rdp->transport->tcp->mac_address; md5 = crypto_md5_init(); crypto_md5_update(md5, mac_address, 6); crypto_md5_final(md5, &license->hwid[HWID_PLATFORM_ID_LENGTH]); } void license_encrypt_premaster_secret(rdpLicense* license) { int key_length; uint8* modulus; uint8* exponent; uint8* encrypted_premaster_secret; exponent = license->certificate->cert_info.exponent; modulus = license->certificate->cert_info.modulus.data; key_length = license->certificate->cert_info.modulus.length; #ifdef WITH_DEBUG_LICENSE printf("modulus (%d bits):\n", key_length * 8); freerdp_hexdump(modulus, key_length); printf("exponent:\n"); freerdp_hexdump(exponent, 4); #endif #if 0 encrypted_premaster_secret = (uint8*) xmalloc(MODULUS_MAX_SIZE); memset(encrypted_premaster_secret, 0, MODULUS_MAX_SIZE); crypto_rsa_encrypt(license->premaster_secret, PREMASTER_SECRET_LENGTH, key_length, modulus, exponent, encrypted_premaster_secret); license->encrypted_premaster_secret->type = BB_ANY_BLOB; license->encrypted_premaster_secret->length = PREMASTER_SECRET_LENGTH; license->encrypted_premaster_secret->data = encrypted_premaster_secret; #else encrypted_premaster_secret = (uint8*) xmalloc(MODULUS_MAX_SIZE); memset(encrypted_premaster_secret, 0, MODULUS_MAX_SIZE); license->encrypted_premaster_secret->type = BB_ANY_BLOB; license->encrypted_premaster_secret->length = MODULUS_MAX_SIZE; license->encrypted_premaster_secret->data = encrypted_premaster_secret; #endif } void license_decrypt_platform_challenge(rdpLicense* license) { CryptoRc4 rc4; license->platform_challenge->data = (uint8*) xmalloc(license->encrypted_platform_challenge->length); license->platform_challenge->length = license->encrypted_platform_challenge->length; rc4 = crypto_rc4_init(license->licensing_encryption_key, LICENSING_ENCRYPTION_KEY_LENGTH); crypto_rc4(rc4, license->encrypted_platform_challenge->length, license->encrypted_platform_challenge->data, license->platform_challenge->data); #ifdef WITH_DEBUG_LICENSE printf("encrypted_platform challenge:\n"); freerdp_hexdump(license->encrypted_platform_challenge->data, license->encrypted_platform_challenge->length); printf("platform challenge:\n"); freerdp_hexdump(license->platform_challenge->data, license->platform_challenge->length); #endif crypto_rc4_free(rc4); } /** * Read Product Information (PRODUCT_INFO).\n * @msdn{cc241915} * @param s stream * @param productInfo product information */ void license_read_product_info(STREAM* s, PRODUCT_INFO* productInfo) { stream_read_uint32(s, productInfo->dwVersion); /* dwVersion (4 bytes) */ stream_read_uint32(s, productInfo->cbCompanyName); /* cbCompanyName (4 bytes) */ productInfo->pbCompanyName = (uint8*) xmalloc(productInfo->cbCompanyName); stream_read(s, productInfo->pbCompanyName, productInfo->cbCompanyName); stream_read_uint32(s, productInfo->cbProductId); /* cbProductId (4 bytes) */ productInfo->pbProductId = (uint8*) xmalloc(productInfo->cbProductId); stream_read(s, productInfo->pbProductId, productInfo->cbProductId); } /** * Allocate New Product Information (PRODUCT_INFO).\n * @msdn{cc241915} * @return new product information */ PRODUCT_INFO* license_new_product_info() { PRODUCT_INFO* productInfo; productInfo = (PRODUCT_INFO*) xmalloc(sizeof(PRODUCT_INFO)); productInfo->dwVersion = 0; productInfo->cbCompanyName = 0; productInfo->pbCompanyName = NULL; productInfo->cbProductId = 0; productInfo->pbProductId = NULL; return productInfo; } /** * Free Product Information (PRODUCT_INFO).\n * @msdn{cc241915} * @param productInfo product information */ void license_free_product_info(PRODUCT_INFO* productInfo) { if (productInfo->pbCompanyName != NULL) xfree(productInfo->pbCompanyName); if (productInfo->pbProductId != NULL) xfree(productInfo->pbProductId); xfree(productInfo); } /** * Read License Binary Blob (LICENSE_BINARY_BLOB).\n * @msdn{cc240481} * @param s stream * @param blob license binary blob */ void license_read_binary_blob(STREAM* s, LICENSE_BLOB* blob) { uint16 wBlobType; stream_read_uint16(s, wBlobType); /* wBlobType (2 bytes) */ if (blob->type != wBlobType && blob->type != BB_ANY_BLOB) { printf("license binary blob type does not match expected type.\n"); return; } blob->type = wBlobType; stream_read_uint16(s, blob->length); /* wBlobLen (2 bytes) */ blob->data = (uint8*) xmalloc(blob->length); stream_read(s, blob->data, blob->length); /* blobData */ } /** * Write License Binary Blob (LICENSE_BINARY_BLOB).\n * @msdn{cc240481} * @param s stream * @param blob license binary blob */ void license_write_binary_blob(STREAM* s, LICENSE_BLOB* blob) { stream_write_uint16(s, blob->type); /* wBlobType (2 bytes) */ stream_write_uint16(s, blob->length); /* wBlobLen (2 bytes) */ if (blob->length > 0) stream_write(s, blob->data, blob->length); /* blobData */ } void license_write_padded_binary_blob(STREAM* s, LICENSE_BLOB* blob) { stream_write_uint16(s, blob->type); /* wBlobType (2 bytes) */ stream_write_uint16(s, blob->length + LICENSING_PADDING_SIZE); /* wBlobLen (2 bytes) */ if (blob->length > 0) stream_write(s, blob->data, blob->length); /* blobData */ stream_write_zero(s, LICENSING_PADDING_SIZE); } /** * Allocate New License Binary Blob (LICENSE_BINARY_BLOB).\n * @msdn{cc240481} * @return new license binary blob */ LICENSE_BLOB* license_new_binary_blob(uint16 type) { LICENSE_BLOB* blob; blob = (LICENSE_BLOB*) xmalloc(sizeof(LICENSE_BLOB)); blob->type = type; blob->length = 0; blob->data = NULL; return blob; } /** * Free License Binary Blob (LICENSE_BINARY_BLOB).\n * @msdn{cc240481} * @param blob license binary blob */ void license_free_binary_blob(LICENSE_BLOB* blob) { if (blob->data != NULL) xfree(blob->data); xfree(blob); } /** * Read License Scope List (SCOPE_LIST).\n * @msdn{cc241916} * @param s stream * @param scopeList scope list */ void license_read_scope_list(STREAM* s, SCOPE_LIST* scopeList) { uint32 i; uint32 scopeCount; stream_read_uint32(s, scopeCount); /* ScopeCount (4 bytes) */ scopeList->count = scopeCount; scopeList->array = (LICENSE_BLOB*) xmalloc(sizeof(LICENSE_BLOB) * scopeCount); /* ScopeArray */ for (i = 0; i < scopeCount; i++) { scopeList->array[i].type = BB_SCOPE_BLOB; license_read_binary_blob(s, &scopeList->array[i]); } } /** * Allocate New License Scope List (SCOPE_LIST).\n * @msdn{cc241916} * @return new scope list */ SCOPE_LIST* license_new_scope_list() { SCOPE_LIST* scopeList; scopeList = (SCOPE_LIST*) xmalloc(sizeof(SCOPE_LIST)); scopeList->count = 0; scopeList->array = NULL; return scopeList; } /** * Free License Scope List (SCOPE_LIST).\n * @msdn{cc241916} * @param scopeList scope list */ void license_free_scope_list(SCOPE_LIST* scopeList) { uint32 i; for (i = 0; i < scopeList->count; i++) { license_free_binary_blob(&scopeList->array[i]); } xfree(scopeList); } /** * Read a LICENSE_REQUEST packet.\n * @msdn{cc241914} * @param license license module * @param s stream */ void license_read_license_request_packet(rdpLicense* license, STREAM* s) { /* ServerRandom (32 bytes) */ stream_read(s, license->server_random, 32); /* ProductInfo */ license_read_product_info(s, license->product_info); /* KeyExchangeList */ license_read_binary_blob(s, license->key_exchange_list); /* ServerCertificate */ license_read_binary_blob(s, license->server_certificate); /* ScopeList */ license_read_scope_list(s, license->scope_list); /* Parse Server Certificate */ certificate_read_server_certificate(license->certificate, license->server_certificate->data, license->server_certificate->length); license_generate_keys(license); license_generate_hwid(license); license_encrypt_premaster_secret(license); } /** * Read a PLATFORM_CHALLENGE packet.\n * @msdn{cc241921} * @param license license module * @param s stream */ void license_read_platform_challenge_packet(rdpLicense* license, STREAM* s) { DEBUG_LICENSE("Receiving Platform Challenge Packet"); stream_seek(s, 4); /* ConnectFlags, Reserved (4 bytes) */ /* EncryptedPlatformChallenge */ license->encrypted_platform_challenge->type = BB_ANY_BLOB; license_read_binary_blob(s, license->encrypted_platform_challenge); license->encrypted_platform_challenge->type = BB_ENCRYPTED_DATA_BLOB; /* MACData (16 bytes) */ stream_seek(s, 16); license_decrypt_platform_challenge(license); } /** * Read a NEW_LICENSE packet.\n * @msdn{cc241926} * @param license license module * @param s stream */ void license_read_new_license_packet(rdpLicense* license, STREAM* s) { DEBUG_LICENSE("Receiving New License Packet"); license->state = LICENSE_STATE_COMPLETED; } /** * Read an UPGRADE_LICENSE packet.\n * @msdn{cc241924} * @param license license module * @param s stream */ void license_read_upgrade_license_packet(rdpLicense* license, STREAM* s) { DEBUG_LICENSE("Receiving Upgrade License Packet"); license->state = LICENSE_STATE_COMPLETED; } /** * Read an ERROR_ALERT packet.\n * @msdn{cc240482} * @param license license module * @param s stream */ void license_read_error_alert_packet(rdpLicense* license, STREAM* s) { uint32 dwErrorCode; uint32 dwStateTransition; stream_read_uint32(s, dwErrorCode); /* dwErrorCode (4 bytes) */ stream_read_uint32(s, dwStateTransition); /* dwStateTransition (4 bytes) */ license_read_binary_blob(s, license->error_info); /* bbErrorInfo */ #ifdef WITH_DEBUG_LICENSE printf("dwErrorCode: %s, dwStateTransition: %s\n", error_codes[dwErrorCode], state_transitions[dwStateTransition]); #endif if (dwErrorCode == STATUS_VALID_CLIENT) { license->state = LICENSE_STATE_COMPLETED; return; } switch (dwStateTransition) { case ST_TOTAL_ABORT: license->state = LICENSE_STATE_ABORTED; break; case ST_NO_TRANSITION: license->state = LICENSE_STATE_COMPLETED; break; case ST_RESET_PHASE_TO_START: license->state = LICENSE_STATE_AWAIT; break; case ST_RESEND_LAST_MESSAGE: break; default: break; } } /** * Write Platform ID.\n * @msdn{cc241918} * @param license license module * @param s stream */ void license_write_platform_id(rdpLicense* license, STREAM* s) { stream_write_uint8(s, 0); /* Client Operating System Version */ stream_write_uint8(s, 0); /* Independent Software Vendor (ISV) */ stream_write_uint16(s, 0); /* Client Software Build */ } /** * Write a NEW_LICENSE_REQUEST packet.\n * @msdn{cc241918} * @param license license module * @param s stream */ void license_write_new_license_request_packet(rdpLicense* license, STREAM* s) { stream_write_uint32(s, KEY_EXCHANGE_ALG_RSA); /* PreferredKeyExchangeAlg (4 bytes) */ license_write_platform_id(license, s); /* PlatformId (4 bytes) */ stream_write(s, license->client_random, 32); /* ClientRandom (32 bytes) */ license_write_padded_binary_blob(s, license->encrypted_premaster_secret); /* EncryptedPremasterSecret */ license_write_binary_blob(s, license->client_user_name); /* ClientUserName */ license_write_binary_blob(s, license->client_machine_name); /* ClientMachineName */ } /** * Send a NEW_LICENSE_REQUEST packet.\n * @msdn{cc241918} * @param license license module */ void license_send_new_license_request_packet(rdpLicense* license) { STREAM* s; s = license_send_stream_init(license); license->client_user_name->data = (uint8*)license->rdp->settings->username; license->client_user_name->length = strlen((char*)license->rdp->settings->username) + 1; license->client_machine_name->data = (uint8*)license->rdp->settings->client_hostname; license->client_machine_name->length = strlen(license->rdp->settings->client_hostname) + 1; license_write_new_license_request_packet(license, s); license_send(license, s, NEW_LICENSE_REQUEST); license->client_user_name->data = NULL; license->client_user_name->length = 0; license->client_machine_name->data = NULL; license->client_machine_name->length = 0; } /** * Write Client Challenge Response Packet.\n * @msdn{cc241922} * @param license license module * @param s stream * @param mac_data signature */ void license_write_platform_challenge_response_packet(rdpLicense* license, STREAM* s, uint8* mac_data) { /* EncryptedPlatformChallengeResponse */ license_write_binary_blob(s, license->encrypted_platform_challenge); /* EncryptedHWID */ license_write_binary_blob(s, license->encrypted_hwid); /* MACData */ stream_write(s, mac_data, 16); } /** * Send Client Challenge Response Packet.\n * @msdn{cc241922} * @param license license module */ void license_send_platform_challenge_response_packet(rdpLicense* license) { STREAM* s; int length; uint8* buffer; CryptoRc4 rc4; uint8 mac_data[16]; s = license_send_stream_init(license); DEBUG_LICENSE("Sending Platform Challenge Response Packet"); license->encrypted_platform_challenge->type = BB_DATA_BLOB; length = license->platform_challenge->length + HWID_LENGTH; buffer = (uint8*) xmalloc(length); memcpy(buffer, license->platform_challenge->data, license->platform_challenge->length); memcpy(&buffer[license->platform_challenge->length], license->hwid, HWID_LENGTH); security_mac_data(license->mac_salt_key, buffer, length, mac_data); xfree(buffer); buffer = (uint8*) xmalloc(HWID_LENGTH); rc4 = crypto_rc4_init(license->licensing_encryption_key, LICENSING_ENCRYPTION_KEY_LENGTH); crypto_rc4(rc4, HWID_LENGTH, license->hwid, buffer); #ifdef WITH_DEBUG_LICENSE printf("Licensing Encryption Key:\n"); freerdp_hexdump(license->licensing_encryption_key, 16); printf("HardwareID:\n"); freerdp_hexdump(license->hwid, 20); printf("Encrypted HardwareID:\n"); freerdp_hexdump(buffer, 20); #endif license->encrypted_hwid->type = BB_DATA_BLOB; license->encrypted_hwid->data = buffer; license->encrypted_hwid->length = HWID_LENGTH; license_write_platform_challenge_response_packet(license, s, mac_data); license_send(license, s, PLATFORM_CHALLENGE_RESPONSE); } /** * Send Server License Error - Valid Client Packet.\n * @msdn{cc241922} * @param license license module */ boolean license_send_valid_client_error_packet(rdpLicense* license) { STREAM* s; s = license_send_stream_init(license); stream_write_uint32(s, STATUS_VALID_CLIENT); /* dwErrorCode */ stream_write_uint32(s, ST_NO_TRANSITION); /* dwStateTransition */ license_write_binary_blob(s, license->error_info); license_send(license, s, ERROR_ALERT); return True; } /** * Instantiate new license module. * @param rdp RDP module * @return new license module */ rdpLicense* license_new(rdpRdp* rdp) { rdpLicense* license; license = (rdpLicense*) xzalloc(sizeof(rdpLicense)); if (license != NULL) { license->rdp = rdp; license->state = LICENSE_STATE_AWAIT; license->certificate = certificate_new(rdp); license->product_info = license_new_product_info(); license->error_info = license_new_binary_blob(BB_ERROR_BLOB); license->key_exchange_list = license_new_binary_blob(BB_KEY_EXCHG_ALG_BLOB); license->server_certificate = license_new_binary_blob(BB_CERTIFICATE_BLOB); license->client_user_name = license_new_binary_blob(BB_CLIENT_USER_NAME_BLOB); license->client_machine_name = license_new_binary_blob(BB_CLIENT_MACHINE_NAME_BLOB); license->platform_challenge = license_new_binary_blob(BB_ANY_BLOB); license->encrypted_platform_challenge = license_new_binary_blob(BB_ANY_BLOB); license->encrypted_premaster_secret = license_new_binary_blob(BB_ANY_BLOB); license->encrypted_hwid = license_new_binary_blob(BB_ENCRYPTED_DATA_BLOB); license->scope_list = license_new_scope_list(); license_generate_randoms(license); } return license; } /** * Free license module. * @param license license module to be freed */ void license_free(rdpLicense* license) { if (license != NULL) { certificate_free(license->certificate); license_free_product_info(license->product_info); license_free_binary_blob(license->error_info); license_free_binary_blob(license->key_exchange_list); license_free_binary_blob(license->server_certificate); license_free_binary_blob(license->client_user_name); license_free_binary_blob(license->client_machine_name); license_free_binary_blob(license->platform_challenge); license_free_binary_blob(license->encrypted_platform_challenge); license_free_binary_blob(license->encrypted_premaster_secret); license_free_binary_blob(license->encrypted_hwid); license_free_scope_list(license->scope_list); xfree(license); } }