mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
601 lines
17 KiB
C++
601 lines
17 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "gmp-test-decryptor.h"
|
|
#include "gmp-test-storage.h"
|
|
#include "gmp-test-output-protection.h"
|
|
|
|
#include <string>
|
|
#include <vector>
|
|
#include <iostream>
|
|
#include <istream>
|
|
#include <iterator>
|
|
#include <sstream>
|
|
#include <set>
|
|
|
|
#include "mozilla/Assertions.h"
|
|
#include "mozilla/Attributes.h"
|
|
#include "mozilla/NullPtr.h"
|
|
|
|
using namespace std;
|
|
|
|
FakeDecryptor* FakeDecryptor::sInstance = nullptr;
|
|
extern GMPPlatformAPI* g_platform_api; // Defined in gmp-fake.cpp
|
|
|
|
class GMPMutexAutoLock
|
|
{
|
|
public:
|
|
explicit GMPMutexAutoLock(GMPMutex* aMutex) : mMutex(aMutex) {
|
|
mMutex->Acquire();
|
|
}
|
|
~GMPMutexAutoLock() {
|
|
mMutex->Release();
|
|
}
|
|
private:
|
|
GMPMutex* const mMutex;
|
|
};
|
|
|
|
class TestManager {
|
|
public:
|
|
TestManager() : mMutex(CreateMutex()) {}
|
|
|
|
// Register a test with the test manager.
|
|
void BeginTest(const string& aTestID) {
|
|
GMPMutexAutoLock lock(mMutex);
|
|
auto found = mTestIDs.find(aTestID);
|
|
if (found == mTestIDs.end()) {
|
|
mTestIDs.insert(aTestID);
|
|
} else {
|
|
Error("FAIL BeginTest test already existed: " + aTestID);
|
|
}
|
|
}
|
|
|
|
// Notify the test manager that the test is finished. If all tests are done,
|
|
// test manager will send "test-storage complete" to notify the parent that
|
|
// all tests are finished and also delete itself.
|
|
void EndTest(const string& aTestID) {
|
|
bool isEmpty = false;
|
|
{
|
|
GMPMutexAutoLock lock(mMutex);
|
|
auto found = mTestIDs.find(aTestID);
|
|
if (found != mTestIDs.end()) {
|
|
mTestIDs.erase(aTestID);
|
|
isEmpty = mTestIDs.empty();
|
|
} else {
|
|
Error("FAIL EndTest test not existed: " + aTestID);
|
|
return;
|
|
}
|
|
}
|
|
if (isEmpty) {
|
|
Finish();
|
|
delete this;
|
|
}
|
|
}
|
|
|
|
private:
|
|
~TestManager() {
|
|
mMutex->Destroy();
|
|
}
|
|
|
|
static void Error(const string& msg) {
|
|
FakeDecryptor::Message(msg);
|
|
}
|
|
|
|
static void Finish() {
|
|
FakeDecryptor::Message("test-storage complete");
|
|
}
|
|
|
|
static GMPMutex* CreateMutex() {
|
|
GMPMutex* mutex = nullptr;
|
|
g_platform_api->createmutex(&mutex);
|
|
return mutex;
|
|
}
|
|
|
|
GMPMutex* const mMutex;
|
|
set<string> mTestIDs;
|
|
};
|
|
|
|
FakeDecryptor::FakeDecryptor(GMPDecryptorHost* aHost)
|
|
: mCallback(nullptr)
|
|
, mHost(aHost)
|
|
{
|
|
MOZ_ASSERT(!sInstance);
|
|
sInstance = this;
|
|
}
|
|
|
|
void FakeDecryptor::DecryptingComplete()
|
|
{
|
|
sInstance = nullptr;
|
|
delete this;
|
|
}
|
|
|
|
void
|
|
FakeDecryptor::Message(const std::string& aMessage)
|
|
{
|
|
MOZ_ASSERT(sInstance);
|
|
const static std::string sid("fake-session-id");
|
|
sInstance->mCallback->SessionMessage(sid.c_str(), sid.size(),
|
|
(const uint8_t*)aMessage.c_str(), aMessage.size(),
|
|
nullptr, 0);
|
|
}
|
|
|
|
std::vector<std::string>
|
|
Tokenize(const std::string& aString)
|
|
{
|
|
std::stringstream strstr(aString);
|
|
std::istream_iterator<std::string> it(strstr), end;
|
|
return std::vector<std::string>(it, end);
|
|
}
|
|
|
|
static const string TruncateRecordId = "truncate-record-id";
|
|
static const string TruncateRecordData = "I will soon be truncated";
|
|
|
|
class ReadThenTask : public GMPTask {
|
|
public:
|
|
ReadThenTask(string aId, ReadContinuation* aThen)
|
|
: mId(aId)
|
|
, mThen(aThen)
|
|
{}
|
|
void Run() MOZ_OVERRIDE {
|
|
ReadRecord(mId, mThen);
|
|
}
|
|
void Destroy() MOZ_OVERRIDE {
|
|
delete this;
|
|
}
|
|
string mId;
|
|
ReadContinuation* mThen;
|
|
};
|
|
|
|
class SendMessageTask : public GMPTask {
|
|
public:
|
|
explicit SendMessageTask(const string& aMessage,
|
|
TestManager* aTestManager = nullptr,
|
|
const string& aTestID = "")
|
|
: mMessage(aMessage), mTestmanager(aTestManager), mTestID(aTestID) {}
|
|
|
|
void Run() MOZ_OVERRIDE {
|
|
FakeDecryptor::Message(mMessage);
|
|
if (mTestmanager) {
|
|
mTestmanager->EndTest(mTestID);
|
|
}
|
|
}
|
|
|
|
void Destroy() MOZ_OVERRIDE {
|
|
delete this;
|
|
}
|
|
|
|
private:
|
|
string mMessage;
|
|
TestManager* const mTestmanager;
|
|
const string mTestID;
|
|
};
|
|
|
|
class TestEmptyContinuation : public ReadContinuation {
|
|
public:
|
|
TestEmptyContinuation(TestManager* aTestManager, const string& aTestID)
|
|
: mTestmanager(aTestManager), mTestID(aTestID) {}
|
|
|
|
void ReadComplete(GMPErr aErr, const std::string& aData) MOZ_OVERRIDE {
|
|
if (aData != "") {
|
|
FakeDecryptor::Message("FAIL TestEmptyContinuation record was not truncated");
|
|
}
|
|
mTestmanager->EndTest(mTestID);
|
|
delete this;
|
|
}
|
|
|
|
private:
|
|
TestManager* const mTestmanager;
|
|
const string mTestID;
|
|
};
|
|
|
|
class TruncateContinuation : public ReadContinuation {
|
|
public:
|
|
TruncateContinuation(const string& aID,
|
|
TestManager* aTestManager,
|
|
const string& aTestID)
|
|
: mID(aID), mTestmanager(aTestManager), mTestID(aTestID) {}
|
|
|
|
void ReadComplete(GMPErr aErr, const std::string& aData) MOZ_OVERRIDE {
|
|
if (aData != TruncateRecordData) {
|
|
FakeDecryptor::Message("FAIL TruncateContinuation read data doesn't match written data");
|
|
}
|
|
auto cont = new TestEmptyContinuation(mTestmanager, mTestID);
|
|
auto msg = "FAIL in TruncateContinuation write.";
|
|
auto failTask = new SendMessageTask(msg, mTestmanager, mTestID);
|
|
WriteRecord(mID, nullptr, 0, new ReadThenTask(mID, cont), failTask);
|
|
delete this;
|
|
}
|
|
|
|
private:
|
|
const string mID;
|
|
TestManager* const mTestmanager;
|
|
const string mTestID;
|
|
};
|
|
|
|
class VerifyAndFinishContinuation : public ReadContinuation {
|
|
public:
|
|
explicit VerifyAndFinishContinuation(string aValue,
|
|
TestManager* aTestManager,
|
|
const string& aTestID)
|
|
: mValue(aValue), mTestmanager(aTestManager), mTestID(aTestID) {}
|
|
|
|
void ReadComplete(GMPErr aErr, const std::string& aData) MOZ_OVERRIDE {
|
|
if (aData != mValue) {
|
|
FakeDecryptor::Message("FAIL VerifyAndFinishContinuation read data doesn't match expected data");
|
|
}
|
|
mTestmanager->EndTest(mTestID);
|
|
delete this;
|
|
}
|
|
|
|
private:
|
|
string mValue;
|
|
TestManager* const mTestmanager;
|
|
const string mTestID;
|
|
};
|
|
|
|
class VerifyAndOverwriteContinuation : public ReadContinuation {
|
|
public:
|
|
VerifyAndOverwriteContinuation(string aId, string aValue, string aOverwrite,
|
|
TestManager* aTestManager, const string& aTestID)
|
|
: mId(aId)
|
|
, mValue(aValue)
|
|
, mOverwrite(aOverwrite)
|
|
, mTestmanager(aTestManager)
|
|
, mTestID(aTestID)
|
|
{}
|
|
|
|
void ReadComplete(GMPErr aErr, const std::string& aData) MOZ_OVERRIDE {
|
|
if (aData != mValue) {
|
|
FakeDecryptor::Message("FAIL VerifyAndOverwriteContinuation read data doesn't match expected data");
|
|
}
|
|
auto cont = new VerifyAndFinishContinuation(mOverwrite, mTestmanager, mTestID);
|
|
auto msg = "FAIL in VerifyAndOverwriteContinuation write.";
|
|
auto failTask = new SendMessageTask(msg, mTestmanager, mTestID);
|
|
WriteRecord(mId, mOverwrite, new ReadThenTask(mId, cont), failTask);
|
|
delete this;
|
|
}
|
|
|
|
private:
|
|
string mId;
|
|
string mValue;
|
|
string mOverwrite;
|
|
TestManager* const mTestmanager;
|
|
const string mTestID;
|
|
};
|
|
|
|
static const string OpenAgainRecordId = "open-again-record-id";
|
|
|
|
class OpenedSecondTimeContinuation : public OpenContinuation {
|
|
public:
|
|
explicit OpenedSecondTimeContinuation(GMPRecord* aRecord,
|
|
TestManager* aTestManager,
|
|
const string& aTestID)
|
|
: mRecord(aRecord), mTestmanager(aTestManager), mTestID(aTestID) {}
|
|
|
|
virtual void OpenComplete(GMPErr aStatus, GMPRecord* aRecord) MOZ_OVERRIDE {
|
|
if (GMP_SUCCEEDED(aStatus)) {
|
|
FakeDecryptor::Message("FAIL OpenSecondTimeContinuation should not be able to re-open record.");
|
|
}
|
|
|
|
// Succeeded, open should have failed.
|
|
mTestmanager->EndTest(mTestID);
|
|
mRecord->Close();
|
|
delete this;
|
|
}
|
|
|
|
private:
|
|
GMPRecord* mRecord;
|
|
TestManager* const mTestmanager;
|
|
const string mTestID;
|
|
};
|
|
|
|
class OpenedFirstTimeContinuation : public OpenContinuation {
|
|
public:
|
|
OpenedFirstTimeContinuation(const string& aID,
|
|
TestManager* aTestManager,
|
|
const string& aTestID)
|
|
: mID(aID), mTestmanager(aTestManager), mTestID(aTestID) {}
|
|
|
|
virtual void OpenComplete(GMPErr aStatus, GMPRecord* aRecord) MOZ_OVERRIDE {
|
|
if (GMP_FAILED(aStatus)) {
|
|
FakeDecryptor::Message("FAIL OpenAgainContinuation to open record initially.");
|
|
mTestmanager->EndTest(mTestID);
|
|
return;
|
|
}
|
|
|
|
auto cont = new OpenedSecondTimeContinuation(aRecord, mTestmanager, mTestID);
|
|
GMPOpenRecord(mID, cont);
|
|
delete this;
|
|
}
|
|
|
|
private:
|
|
const string mID;
|
|
TestManager* const mTestmanager;
|
|
const string mTestID;
|
|
};
|
|
|
|
static void
|
|
DoTestStorage(const string& aPrefix, TestManager* aTestManager)
|
|
{
|
|
// Basic I/O tests. We run three cases concurrently. The tests, like
|
|
// GMPStorage run asynchronously. When they've all passed, we send
|
|
// a message back to the parent process, or a failure message if not.
|
|
|
|
// Test 1: Basic I/O test, and test that writing 0 bytes in a record
|
|
// deletes record.
|
|
//
|
|
// Write data to truncate record, then
|
|
// read data, verify that we read what we wrote, then
|
|
// write 0 bytes to truncate record, then
|
|
// read data, verify that 0 bytes was read
|
|
const string id1 = aPrefix + TruncateRecordId;
|
|
const string testID1 = aPrefix + "write-test-1";
|
|
aTestManager->BeginTest(testID1);
|
|
auto cont1 = new TruncateContinuation(id1, aTestManager, testID1);
|
|
auto msg1 = "FAIL in TestStorage writing TruncateRecord.";
|
|
auto failTask1 = new SendMessageTask(msg1, aTestManager, testID1);
|
|
WriteRecord(id1, TruncateRecordData,
|
|
new ReadThenTask(id1, cont1), failTask1);
|
|
|
|
// Test 2: Test that overwriting a record with a shorter record truncates
|
|
// the record to the shorter record.
|
|
//
|
|
// Write record, then
|
|
// read and verify record, then
|
|
// write a shorter record to same record.
|
|
// read and verify
|
|
string id2 = aPrefix + "record1";
|
|
string record1 = "This is the first write to a record.";
|
|
string overwrite = "A shorter record";
|
|
const string testID2 = aPrefix + "write-test-2";
|
|
aTestManager->BeginTest(testID2);
|
|
auto task2 = new VerifyAndOverwriteContinuation(id2, record1, overwrite,
|
|
aTestManager, testID2);
|
|
auto msg2 = "FAIL in TestStorage writing record1.";
|
|
auto failTask2 = new SendMessageTask(msg2, aTestManager, testID2);
|
|
WriteRecord(id2, record1, new ReadThenTask(id2, task2), failTask2);
|
|
|
|
// Test 3: Test that opening a record while it's already open fails.
|
|
//
|
|
// Open record1, then
|
|
// open record1, should fail.
|
|
// close record1
|
|
const string id3 = aPrefix + OpenAgainRecordId;
|
|
const string testID3 = aPrefix + "open-test-1";
|
|
aTestManager->BeginTest(testID3);
|
|
auto task3 = new OpenedFirstTimeContinuation(id3, aTestManager, testID3);
|
|
GMPOpenRecord(id3, task3);
|
|
}
|
|
|
|
class TestStorageTask : public GMPTask {
|
|
public:
|
|
TestStorageTask(const string& aPrefix, TestManager* aTestManager)
|
|
: mPrefix(aPrefix), mTestManager(aTestManager) {}
|
|
virtual void Destroy() { delete this; }
|
|
virtual void Run() {
|
|
DoTestStorage(mPrefix, mTestManager);
|
|
}
|
|
private:
|
|
const string mPrefix;
|
|
TestManager* const mTestManager;
|
|
};
|
|
|
|
void
|
|
FakeDecryptor::TestStorage()
|
|
{
|
|
TestManager* testManager = new TestManager();
|
|
GMPThread* thread1 = nullptr;
|
|
GMPThread* thread2 = nullptr;
|
|
|
|
// Main thread tests.
|
|
DoTestStorage("mt1-", testManager);
|
|
DoTestStorage("mt2-", testManager);
|
|
|
|
// Off-main-thread tests.
|
|
if (GMP_SUCCEEDED(g_platform_api->createthread(&thread1))) {
|
|
thread1->Post(new TestStorageTask("thread1-", testManager));
|
|
} else {
|
|
FakeDecryptor::Message("FAIL to create thread1 for storage tests");
|
|
}
|
|
|
|
if (GMP_SUCCEEDED(g_platform_api->createthread(&thread2))) {
|
|
thread2->Post(new TestStorageTask("thread2-", testManager));
|
|
} else {
|
|
FakeDecryptor::Message("FAIL to create thread2 for storage tests");
|
|
}
|
|
|
|
if (thread1) {
|
|
thread1->Join();
|
|
}
|
|
|
|
if (thread2) {
|
|
thread2->Join();
|
|
}
|
|
|
|
// Note: Once all tests finish, TestManager will dispatch "test-pass" message,
|
|
// which ends the test for the parent.
|
|
}
|
|
|
|
class ReportWritten : public GMPTask {
|
|
public:
|
|
ReportWritten(const string& aRecordId, const string& aValue)
|
|
: mRecordId(aRecordId)
|
|
, mValue(aValue)
|
|
{}
|
|
void Run() MOZ_OVERRIDE {
|
|
FakeDecryptor::Message("stored " + mRecordId + " " + mValue);
|
|
}
|
|
void Destroy() MOZ_OVERRIDE {
|
|
delete this;
|
|
}
|
|
const string mRecordId;
|
|
const string mValue;
|
|
};
|
|
|
|
class ReportReadStatusContinuation : public ReadContinuation {
|
|
public:
|
|
explicit ReportReadStatusContinuation(const string& aRecordId)
|
|
: mRecordId(aRecordId)
|
|
{}
|
|
void ReadComplete(GMPErr aErr, const std::string& aData) MOZ_OVERRIDE {
|
|
if (GMP_FAILED(aErr)) {
|
|
FakeDecryptor::Message("retrieve " + mRecordId + " failed");
|
|
} else {
|
|
stringstream ss;
|
|
ss << aData.size();
|
|
string len;
|
|
ss >> len;
|
|
FakeDecryptor::Message("retrieve " + mRecordId + " succeeded (length " +
|
|
len + " bytes)");
|
|
}
|
|
delete this;
|
|
}
|
|
string mRecordId;
|
|
};
|
|
|
|
class ReportReadRecordContinuation : public ReadContinuation {
|
|
public:
|
|
explicit ReportReadRecordContinuation(const string& aRecordId)
|
|
: mRecordId(aRecordId)
|
|
{}
|
|
void ReadComplete(GMPErr aErr, const std::string& aData) MOZ_OVERRIDE {
|
|
if (GMP_FAILED(aErr)) {
|
|
FakeDecryptor::Message("retrieved " + mRecordId + " failed");
|
|
} else {
|
|
FakeDecryptor::Message("retrieved " + mRecordId + " " + aData);
|
|
}
|
|
delete this;
|
|
}
|
|
string mRecordId;
|
|
};
|
|
|
|
static void
|
|
RecvGMPRecordIterator(GMPRecordIterator* aRecordIterator,
|
|
void* aUserArg,
|
|
GMPErr aStatus)
|
|
{
|
|
FakeDecryptor* decryptor = reinterpret_cast<FakeDecryptor*>(aUserArg);
|
|
decryptor->ProcessRecordNames(aRecordIterator, aStatus);
|
|
}
|
|
|
|
void
|
|
FakeDecryptor::ProcessRecordNames(GMPRecordIterator* aRecordIterator,
|
|
GMPErr aStatus)
|
|
{
|
|
if (sInstance != this) {
|
|
FakeDecryptor::Message("Error aUserArg was not passed through GetRecordIterator");
|
|
return;
|
|
}
|
|
if (GMP_FAILED(aStatus)) {
|
|
FakeDecryptor::Message("Error GetRecordIterator failed");
|
|
return;
|
|
}
|
|
std::string response("record-names ");
|
|
bool first = true;
|
|
const char* name = nullptr;
|
|
uint32_t len = 0;
|
|
while (GMP_SUCCEEDED(aRecordIterator->GetName(&name, &len))) {
|
|
std::string s(name, name+len);
|
|
if (!first) {
|
|
response += ",";
|
|
} else {
|
|
first = false;
|
|
}
|
|
response += s;
|
|
aRecordIterator->NextRecord();
|
|
}
|
|
aRecordIterator->Close();
|
|
FakeDecryptor::Message(response);
|
|
}
|
|
|
|
enum ShutdownMode {
|
|
ShutdownNormal,
|
|
ShutdownTimeout,
|
|
ShutdownStoreToken
|
|
};
|
|
|
|
static ShutdownMode sShutdownMode = ShutdownNormal;
|
|
static string sShutdownToken = "";
|
|
|
|
void
|
|
FakeDecryptor::UpdateSession(uint32_t aPromiseId,
|
|
const char* aSessionId,
|
|
uint32_t aSessionIdLength,
|
|
const uint8_t* aResponse,
|
|
uint32_t aResponseSize)
|
|
{
|
|
std::string response((const char*)aResponse, (const char*)(aResponse)+aResponseSize);
|
|
std::vector<std::string> tokens = Tokenize(response);
|
|
const string& task = tokens[0];
|
|
if (task == "test-storage") {
|
|
TestStorage();
|
|
} else if (task == "store") {
|
|
// send "stored record" message on complete.
|
|
const string& id = tokens[1];
|
|
const string& value = tokens[2];
|
|
WriteRecord(id,
|
|
value,
|
|
new ReportWritten(id, value),
|
|
new SendMessageTask("FAIL in writing record."));
|
|
} else if (task == "retrieve") {
|
|
const string& id = tokens[1];
|
|
ReadRecord(id, new ReportReadStatusContinuation(id));
|
|
} else if (task == "shutdown-mode") {
|
|
const string& mode = tokens[1];
|
|
if (mode == "timeout") {
|
|
sShutdownMode = ShutdownTimeout;
|
|
} else if (mode == "token") {
|
|
sShutdownMode = ShutdownStoreToken;
|
|
sShutdownToken = tokens[2];
|
|
Message("shutdown-token received " + sShutdownToken);
|
|
}
|
|
} else if (task == "retrieve-shutdown-token") {
|
|
ReadRecord("shutdown-token", new ReportReadRecordContinuation("shutdown-token"));
|
|
} else if (task == "test-op-apis") {
|
|
mozilla::gmptest::TestOuputProtectionAPIs();
|
|
} else if (task == "retrieve-plugin-voucher") {
|
|
const uint8_t* rawVoucher = nullptr;
|
|
uint32_t length = 0;
|
|
mHost->GetPluginVoucher(&rawVoucher, &length);
|
|
std::string voucher((const char*)rawVoucher, (const char*)(rawVoucher + length));
|
|
Message("retrieved plugin-voucher: " + voucher);
|
|
} else if (task == "retrieve-record-names") {
|
|
GMPEnumRecordNames(&RecvGMPRecordIterator, this);
|
|
}
|
|
}
|
|
|
|
class CompleteShutdownTask : public GMPTask {
|
|
public:
|
|
explicit CompleteShutdownTask(GMPAsyncShutdownHost* aHost)
|
|
: mHost(aHost)
|
|
{
|
|
}
|
|
virtual void Run() {
|
|
mHost->ShutdownComplete();
|
|
}
|
|
virtual void Destroy() { delete this; }
|
|
GMPAsyncShutdownHost* mHost;
|
|
};
|
|
|
|
void
|
|
TestAsyncShutdown::BeginShutdown() {
|
|
switch (sShutdownMode) {
|
|
case ShutdownNormal:
|
|
mHost->ShutdownComplete();
|
|
break;
|
|
case ShutdownTimeout:
|
|
// Don't do anything; wait for timeout, Gecko should kill
|
|
// the plugin and recover.
|
|
break;
|
|
case ShutdownStoreToken:
|
|
// Store message, then shutdown.
|
|
WriteRecord("shutdown-token",
|
|
sShutdownToken,
|
|
new CompleteShutdownTask(mHost),
|
|
new SendMessageTask("FAIL writing shutdown-token."));
|
|
break;
|
|
}
|
|
}
|