/* * Copyright (c) 2013 Matt Corallo */ #include "curve25519-donna.h" #include #include #include #include #include #include const unsigned char basepoint[32] = {9}; class Curve25519Instance : public pp::Instance { public: explicit Curve25519Instance(PP_Instance instance) : pp::Instance(instance) {} virtual void HandleMessage(const pp::Var& var_message) { if (!var_message.is_dictionary()) return; // Go away broken client pp::VarDictionary dictionary(var_message); pp::VarArrayBuffer privArrBuff(dictionary.Get("priv")); if (privArrBuff.is_null() || privArrBuff.ByteLength() != 32) return; // Go away broken client unsigned char* priv = static_cast(privArrBuff.Map()); pp::VarArrayBuffer resBuffer(32); unsigned char* res = static_cast(resBuffer.Map()); std::string command = dictionary.Get("command").AsString(); if (command == "bytesToPriv") { memcpy(res, priv, 32); res[0] &= 248; res[31] &= 127; res[31] |= 64; } else if (command == "privToPub") { curve25519_donna(res, priv, basepoint); } else if (command == "ECDHE") { pp::VarArrayBuffer pubArrBuff(dictionary.Get("pub")); if (!pubArrBuff.is_null() && pubArrBuff.ByteLength() == 32) { unsigned char* pub = static_cast(pubArrBuff.Map()); curve25519_donna(res, priv, pub); pubArrBuff.Unmap(); } } resBuffer.Unmap(); privArrBuff.Unmap(); pp::VarDictionary returnMessage; returnMessage.Set("call_id", dictionary.Get("call_id").AsInt()); returnMessage.Set("res", resBuffer); PostMessage(returnMessage); } }; class Curve25519Module : public pp::Module { public: Curve25519Module() : pp::Module() {} virtual pp::Instance* CreateInstance(PP_Instance instance) { return new Curve25519Instance(instance); } }; namespace pp { Module* CreateModule() { return new Curve25519Module(); } }