#include "nvpairingmanager.h" #include "utils.h" #include #include #include #include #include #include #define REQUEST_TIMEOUT_MS 5000 NvPairingManager::NvPairingManager(NvComputer* computer) : m_Http(computer) { QByteArray cert = IdentityManager::get()->getCertificate(); BIO *bio = BIO_new_mem_buf(cert.data(), -1); THROW_BAD_ALLOC_IF_NULL(bio); m_Cert = PEM_read_bio_X509(bio, nullptr, nullptr, nullptr); BIO_free_all(bio); if (m_Cert == nullptr) { throw std::runtime_error("Unable to load certificate"); } QByteArray pk = IdentityManager::get()->getPrivateKey(); bio = BIO_new_mem_buf(pk.data(), -1); THROW_BAD_ALLOC_IF_NULL(bio); m_PrivateKey = PEM_read_bio_PrivateKey(bio, nullptr, nullptr, nullptr); BIO_free_all(bio); if (m_PrivateKey == nullptr) { throw std::runtime_error("Unable to load private key"); } } NvPairingManager::~NvPairingManager() { X509_free(m_Cert); EVP_PKEY_free(m_PrivateKey); } QByteArray NvPairingManager::generateRandomBytes(int length) { char* data = static_cast(alloca(length)); RAND_bytes(reinterpret_cast(data), length); return QByteArray(data, length); } QByteArray NvPairingManager::encrypt(const QByteArray& plaintext, const QByteArray& key) { QByteArray ciphertext(plaintext.size(), 0); EVP_CIPHER_CTX* cipher; int ciphertextLen; cipher = EVP_CIPHER_CTX_new(); THROW_BAD_ALLOC_IF_NULL(cipher); EVP_EncryptInit(cipher, EVP_aes_128_ecb(), reinterpret_cast(key.data()), NULL); EVP_CIPHER_CTX_set_padding(cipher, 0); EVP_EncryptUpdate(cipher, reinterpret_cast(ciphertext.data()), &ciphertextLen, reinterpret_cast(plaintext.data()), plaintext.length()); Q_ASSERT(ciphertextLen == ciphertext.length()); EVP_CIPHER_CTX_free(cipher); return ciphertext; } QByteArray NvPairingManager::decrypt(const QByteArray& ciphertext, const QByteArray& key) { QByteArray plaintext(ciphertext.size(), 0); EVP_CIPHER_CTX* cipher; int plaintextLen; cipher = EVP_CIPHER_CTX_new(); THROW_BAD_ALLOC_IF_NULL(cipher); EVP_DecryptInit(cipher, EVP_aes_128_ecb(), reinterpret_cast(key.data()), NULL); EVP_CIPHER_CTX_set_padding(cipher, 0); EVP_DecryptUpdate(cipher, reinterpret_cast(plaintext.data()), &plaintextLen, reinterpret_cast(ciphertext.data()), ciphertext.length()); Q_ASSERT(plaintextLen == plaintext.length()); EVP_CIPHER_CTX_free(cipher); return plaintext; } QByteArray NvPairingManager::getSignatureFromPemCert(const QByteArray& certificate) { #if (OPENSSL_VERSION_NUMBER < 0x10100000L) BIO* bio = BIO_new_mem_buf(const_cast(certificate.data()), -1); #else BIO* bio = BIO_new_mem_buf(certificate.data(), -1); #endif THROW_BAD_ALLOC_IF_NULL(bio); X509* cert = PEM_read_bio_X509(bio, nullptr, nullptr, nullptr); BIO_free_all(bio); #if (OPENSSL_VERSION_NUMBER < 0x10002000L) ASN1_BIT_STRING *asnSignature = cert->signature; #elif (OPENSSL_VERSION_NUMBER < 0x10100000L) ASN1_BIT_STRING *asnSignature; X509_get0_signature(&asnSignature, NULL, cert); #else const ASN1_BIT_STRING *asnSignature; X509_get0_signature(&asnSignature, NULL, cert); #endif QByteArray signature(reinterpret_cast(asnSignature->data), asnSignature->length); X509_free(cert); return signature; } bool NvPairingManager::verifySignature(const QByteArray& data, const QByteArray& signature, const QByteArray& serverCertificate) { #if (OPENSSL_VERSION_NUMBER < 0x10100000L) BIO* bio = BIO_new_mem_buf(const_cast(serverCertificate.data()), -1); #else BIO* bio = BIO_new_mem_buf(serverCertificate.data(), -1); #endif THROW_BAD_ALLOC_IF_NULL(bio); X509* cert = PEM_read_bio_X509(bio, nullptr, nullptr, nullptr); BIO_free_all(bio); EVP_PKEY* pubKey = X509_get_pubkey(cert); THROW_BAD_ALLOC_IF_NULL(pubKey); EVP_MD_CTX* mdctx = EVP_MD_CTX_create(); THROW_BAD_ALLOC_IF_NULL(mdctx); EVP_DigestVerifyInit(mdctx, nullptr, EVP_sha256(), nullptr, pubKey); EVP_DigestVerifyUpdate(mdctx, data.data(), data.length()); int result = EVP_DigestVerifyFinal(mdctx, reinterpret_cast(const_cast(signature.data())), signature.length()); EVP_PKEY_free(pubKey); EVP_MD_CTX_destroy(mdctx); X509_free(cert); return result > 0; } QByteArray NvPairingManager::signMessage(const QByteArray& message) { EVP_MD_CTX *ctx = EVP_MD_CTX_create(); THROW_BAD_ALLOC_IF_NULL(ctx); EVP_DigestSignInit(ctx, NULL, EVP_sha256(), NULL, m_PrivateKey); EVP_DigestSignUpdate(ctx, reinterpret_cast(const_cast(message.data())), message.length()); size_t signatureLength = 0; EVP_DigestSignFinal(ctx, NULL, &signatureLength); QByteArray signature((int)signatureLength, 0); EVP_DigestSignFinal(ctx, reinterpret_cast(signature.data()), &signatureLength); EVP_MD_CTX_destroy(ctx); return signature; } QByteArray NvPairingManager::saltPin(const QByteArray& salt, QString pin) { return QByteArray().append(salt).append(pin.toLatin1()); } NvPairingManager::PairState NvPairingManager::pair(QString appVersion, QString pin, QSslCertificate& serverCert) { int serverMajorVersion = NvHTTP::parseQuad(appVersion).at(0); qInfo() << "Pairing with server generation:" << serverMajorVersion; QCryptographicHash::Algorithm hashAlgo; int hashLength; if (serverMajorVersion >= 7) { // Gen 7+ uses SHA-256 hashing hashAlgo = QCryptographicHash::Sha256; hashLength = 32; } else { // Prior to Gen 7 uses SHA-1 hashing hashAlgo = QCryptographicHash::Sha1; hashLength = 20; } QByteArray salt = generateRandomBytes(16); QByteArray saltedPin = saltPin(salt, pin); QByteArray aesKey = QCryptographicHash::hash(saltedPin, hashAlgo).constData(); aesKey.truncate(16); QString getCert = m_Http.openConnectionToString(m_Http.m_BaseUrlHttp, "pair", "devicename=roth&updateState=1&phrase=getservercert&salt=" + salt.toHex() + "&clientcert=" + IdentityManager::get()->getCertificate().toHex(), 0); NvHTTP::verifyResponseStatus(getCert); if (NvHTTP::getXmlString(getCert, "paired") != "1") { qCritical() << "Failed pairing at stage #1"; return PairState::FAILED; } QByteArray serverCertStr = NvHTTP::getXmlStringFromHex(getCert, "plaincert"); if (serverCertStr == nullptr) { qCritical() << "Server likely already pairing"; m_Http.openConnectionToString(m_Http.m_BaseUrlHttp, "unpair", nullptr, REQUEST_TIMEOUT_MS); return PairState::ALREADY_IN_PROGRESS; } QSslCertificate unverifiedServerCert = QSslCertificate(serverCertStr); if (unverifiedServerCert.isNull()) { Q_ASSERT(!unverifiedServerCert.isNull()); qCritical() << "Failed to parse plaincert"; m_Http.openConnectionToString(m_Http.m_BaseUrlHttp, "unpair", nullptr, REQUEST_TIMEOUT_MS); return PairState::FAILED; } // Pin this cert for TLS until pairing is complete. If successful, we will propagate // the cert into the NvComputer object and persist it. m_Http.setServerCert(unverifiedServerCert); QByteArray randomChallenge = generateRandomBytes(16); QByteArray encryptedChallenge = encrypt(randomChallenge, aesKey); QString challengeXml = m_Http.openConnectionToString(m_Http.m_BaseUrlHttp, "pair", "devicename=roth&updateState=1&clientchallenge=" + encryptedChallenge.toHex(), REQUEST_TIMEOUT_MS); NvHTTP::verifyResponseStatus(challengeXml); if (NvHTTP::getXmlString(challengeXml, "paired") != "1") { qCritical() << "Failed pairing at stage #2"; m_Http.openConnectionToString(m_Http.m_BaseUrlHttp, "unpair", nullptr, REQUEST_TIMEOUT_MS); return PairState::FAILED; } QByteArray challengeResponseData = decrypt(m_Http.getXmlStringFromHex(challengeXml, "challengeresponse"), aesKey); QByteArray clientSecretData = generateRandomBytes(16); QByteArray challengeResponse; QByteArray serverResponse(challengeResponseData.data(), hashLength); #if (OPENSSL_VERSION_NUMBER < 0x10002000L) ASN1_BIT_STRING *asnSignature = m_Cert->signature; #elif (OPENSSL_VERSION_NUMBER < 0x10100000L) ASN1_BIT_STRING *asnSignature; X509_get0_signature(&asnSignature, NULL, m_Cert); #else const ASN1_BIT_STRING *asnSignature; X509_get0_signature(&asnSignature, NULL, m_Cert); #endif challengeResponse.append(challengeResponseData.data() + hashLength, 16); challengeResponse.append(reinterpret_cast(asnSignature->data), asnSignature->length); challengeResponse.append(clientSecretData); QByteArray paddedHash = QCryptographicHash::hash(challengeResponse, hashAlgo); paddedHash.resize(32); QByteArray encryptedChallengeResponseHash = encrypt(paddedHash, aesKey); QString respXml = m_Http.openConnectionToString(m_Http.m_BaseUrlHttp, "pair", "devicename=roth&updateState=1&serverchallengeresp=" + encryptedChallengeResponseHash.toHex(), REQUEST_TIMEOUT_MS); NvHTTP::verifyResponseStatus(respXml); if (NvHTTP::getXmlString(respXml, "paired") != "1") { qCritical() << "Failed pairing at stage #3"; m_Http.openConnectionToString(m_Http.m_BaseUrlHttp, "unpair", nullptr, REQUEST_TIMEOUT_MS); return PairState::FAILED; } QByteArray pairingSecret = NvHTTP::getXmlStringFromHex(respXml, "pairingsecret"); QByteArray serverSecret = pairingSecret.left(16); QByteArray serverSignature = pairingSecret.mid(16); if (!verifySignature(serverSecret, serverSignature, serverCertStr)) { qCritical() << "MITM detected"; m_Http.openConnectionToString(m_Http.m_BaseUrlHttp, "unpair", nullptr, REQUEST_TIMEOUT_MS); return PairState::FAILED; } QByteArray expectedResponseData; expectedResponseData.append(randomChallenge); expectedResponseData.append(getSignatureFromPemCert(serverCertStr)); expectedResponseData.append(serverSecret); if (QCryptographicHash::hash(expectedResponseData, hashAlgo) != serverResponse) { qCritical() << "Incorrect PIN"; m_Http.openConnectionToString(m_Http.m_BaseUrlHttp, "unpair", nullptr, REQUEST_TIMEOUT_MS); return PairState::PIN_WRONG; } QByteArray clientPairingSecret; clientPairingSecret.append(clientSecretData); clientPairingSecret.append(signMessage(clientSecretData)); QString secretRespXml = m_Http.openConnectionToString(m_Http.m_BaseUrlHttp, "pair", "devicename=roth&updateState=1&clientpairingsecret=" + clientPairingSecret.toHex(), REQUEST_TIMEOUT_MS); NvHTTP::verifyResponseStatus(secretRespXml); if (NvHTTP::getXmlString(secretRespXml, "paired") != "1") { qCritical() << "Failed pairing at stage #4"; m_Http.openConnectionToString(m_Http.m_BaseUrlHttp, "unpair", nullptr, REQUEST_TIMEOUT_MS); return PairState::FAILED; } QString pairChallengeXml = m_Http.openConnectionToString(m_Http.m_BaseUrlHttps, "pair", "devicename=roth&updateState=1&phrase=pairchallenge", REQUEST_TIMEOUT_MS); NvHTTP::verifyResponseStatus(pairChallengeXml); if (NvHTTP::getXmlString(pairChallengeXml, "paired") != "1") { qCritical() << "Failed pairing at stage #5"; m_Http.openConnectionToString(m_Http.m_BaseUrlHttp, "unpair", nullptr, REQUEST_TIMEOUT_MS); return PairState::FAILED; } serverCert = std::move(unverifiedServerCert); return PairState::PAIRED; }