Home | History | Annotate | Download | only in cpp
      1 /**
      2  * Copyright (C) 2010 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 #include <queue>
     18 #include <v8.h>
     19 
     20 #include "logging.h"
     21 #include "js_support.h"
     22 #include "node_object_wrap.h"
     23 #include "node_util.h"
     24 #include "util.h"
     25 #include "worker.h"
     26 
     27 #include "worker_v8.h"
     28 
     29 
     30 //#define WORKER_V8_V8_DEBUG
     31 #ifdef  WORKER_V8_V8_DEBUG
     32 
     33 #define DBG(...) LOGD(__VA_ARGS__)
     34 
     35 #else
     36 
     37 #define DBG(...)
     38 
     39 #endif
     40 
     41 v8::Persistent<v8::FunctionTemplate> WorkerV8Template;
     42 
     43 class WorkerV8 : public ObjectWrap {
     44   private:
     45     friend class Handler;
     46 
     47 
     48     struct ArgInfo {
     49         v8::Persistent<v8::Object> js_this;
     50         v8::Persistent<v8::Value> value;
     51     };
     52 
     53     pthread_mutex_t ai_free_list_mutex_;
     54     std::queue<ArgInfo *>  ai_free_list_;
     55 
     56     ArgInfo *ObtainArgInfo() {
     57         ArgInfo *ai;
     58         pthread_mutex_lock(&ai_free_list_mutex_);
     59         if (ai_free_list_.size() == 0) {
     60             ai = new ArgInfo();
     61         } else {
     62             ai = ai_free_list_.front();
     63             ai_free_list_.pop();
     64         }
     65         pthread_mutex_unlock(&ai_free_list_mutex_);
     66         return ai;
     67     }
     68 
     69     void ReleaseArgInfo(ArgInfo *ai) {
     70         pthread_mutex_lock(&ai_free_list_mutex_);
     71         ai_free_list_.push(ai);
     72         pthread_mutex_unlock(&ai_free_list_mutex_);
     73     }
     74 
     75     class Handler : public WorkerQueue {
     76       private:
     77         v8::Persistent<v8::Value> functionValue_;
     78         WorkerV8 *worker_;
     79 
     80       public:
     81         Handler(WorkerV8 *worker, v8::Handle<v8::Value> value)
     82             : worker_(worker) {
     83             functionValue_ = v8::Persistent<v8::Value>::New(value);
     84         }
     85 
     86         void Process(void *param) {
     87             DBG("Handler::Process: E");
     88 
     89             v8::Locker locker;
     90             v8::HandleScope handle_scope;
     91             v8::TryCatch try_catch;
     92             try_catch.SetVerbose(true);
     93 
     94             ArgInfo *ai = (ArgInfo*)param;
     95             v8::Handle<v8::Value> args(ai->value);
     96             v8::Function::Cast(*functionValue_)->Call(ai->js_this, 1, &args);
     97 
     98             ai->js_this.Dispose();
     99             ai->value.Dispose();
    100 
    101             worker_->ReleaseArgInfo(ai);
    102 
    103             DBG("Handler::Process: X");
    104         }
    105     };
    106 
    107     Handler *handler_;
    108 
    109   public:
    110     WorkerV8(v8::Handle<v8::Object> self, v8::Handle<v8::Value> functionValue) {
    111         DBG("WorkerV8::WorkerV8 E:");
    112         pthread_mutex_init(&ai_free_list_mutex_, NULL);
    113         handler_ = new Handler(this, functionValue);
    114         Wrap(self);
    115         DBG("WorkerV8::WorkerV8 X: this=%p handler_=%p", this, handler_);
    116     }
    117 
    118     virtual ~WorkerV8() {
    119         DBG("~WorkerV8::WorkerV8 E:");
    120         DBG("~WorkerV8::WorkerV8 X:");
    121     }
    122 
    123     static v8::Handle<v8::Value> Run(const v8::Arguments& args) {
    124         WorkerV8 *workerV8 = ObjectWrap::Unwrap<WorkerV8>(args.This());
    125         DBG("WorkerV8::Run(args) E:");
    126         workerV8->handler_->Run();
    127         DBG("WorkerV8::Run(args) X:");
    128         return v8::Undefined();
    129     }
    130 
    131     static v8::Handle<v8::Value> Add(const v8::Arguments& args) {
    132         DBG("WorkerV8::Add(args) E:");
    133         WorkerV8 *workerV8 = ObjectWrap::Unwrap<WorkerV8>(args.This());
    134 
    135         // Validate one argument to add
    136         if (args.Length() != 1) {
    137             DBG("WorkerV8::Add(args) X: expecting one param");
    138             return v8::ThrowException(v8::String::New("Add has no parameter"));
    139         }
    140         ArgInfo *ai = workerV8->ObtainArgInfo();
    141         ai->js_this = v8::Persistent<v8::Object>::New( args.This() );
    142         ai->value = v8::Persistent<v8::Value>::New( args[0] );
    143 
    144         workerV8->handler_->Add(ai);
    145         DBG("WorkerV8::Add(args) X:");
    146         return v8::Undefined();
    147     }
    148 
    149     static v8::Handle<v8::Value> AddDelayed(const v8::Arguments& args) {
    150         DBG("WorkerV8::AddDelayed(args) E:");
    151         WorkerV8 *workerV8 = ObjectWrap::Unwrap<WorkerV8>(args.This());
    152 
    153         // Validate two argument to addDelayed
    154         if (args.Length() != 2) {
    155             DBG("WorkerV8::AddDelayed(args) X: expecting two params");
    156             return v8::ThrowException(v8::String::New("AddDelayed expects req delayTime params"));
    157         }
    158         ArgInfo *ai = workerV8->ObtainArgInfo();
    159         ai->js_this = v8::Persistent<v8::Object>::New( args.This() );
    160         ai->value = v8::Persistent<v8::Value>::New( args[0] );
    161         v8::Handle<v8::Value> v8DelayMs(args[1]->ToObject());
    162         int32_t delay_ms = v8DelayMs->Int32Value();
    163         workerV8->handler_->AddDelayed(ai, delay_ms);
    164 
    165         DBG("WorkerV8::AddDelayed(args) X:");
    166         return v8::Undefined();
    167     }
    168 
    169     static v8::Handle<v8::Value> NewWorkerV8(const v8::Arguments& args) {
    170         DBG("WorkerV8::NewWorkerV8 E: args.Length()=%d", args.Length());
    171         WorkerV8 *worker = new WorkerV8(args.This(), args[0]);
    172         DBG("WorkerV8::NewWorkerV8 X:");
    173         return worker->handle_;
    174     }
    175 };
    176 
    177 void WorkerV8Init() {
    178     DBG("WorkerV8Init E:");
    179     v8::HandleScope handle_scope;
    180 
    181     WorkerV8Template = v8::Persistent<v8::FunctionTemplate>::New(
    182                            v8::FunctionTemplate::New(WorkerV8::NewWorkerV8));
    183     WorkerV8Template->SetClassName(v8::String::New("Worker"));
    184     // native self (Field 0 is handle_) field count is at least 1
    185     WorkerV8Template->InstanceTemplate()->SetInternalFieldCount(1);
    186 
    187     // Set prototype methods
    188     SET_PROTOTYPE_METHOD(WorkerV8Template, "run", WorkerV8::Run);
    189     SET_PROTOTYPE_METHOD(WorkerV8Template, "add", WorkerV8::Add);
    190     SET_PROTOTYPE_METHOD(WorkerV8Template, "addDelayed", WorkerV8::AddDelayed);
    191 
    192     DBG("WorkerV8Init X:");
    193 }
    194 
    195 void testWorkerV8(v8::Handle<v8::Context> context) {
    196     LOGD("testWorkerV8 E: ********");
    197     v8::HandleScope handle_scope;
    198 
    199     v8::TryCatch try_catch;
    200     try_catch.SetVerbose(true);
    201 
    202     LOGD("testWorkerV8 runJs");
    203     runJs(context, &try_catch, "local-string",
    204         "var w1 = new Worker(function (msg) {"
    205         "     print('w1: ' + msg);\n"
    206         "});\n"
    207         "w1.run();\n"
    208         "var w2 = new Worker(function (msg) {"
    209         "     print('w2: ' + msg);\n"
    210         "});\n"
    211         "w2.run();\n"
    212         "w2.addDelayed('three', 1000);\n"
    213         "w2.add('one');\n"
    214         "w1.add('two');\n"
    215         "w1.addDelayed('four', 2000);\n"
    216     );
    217     LOGD("testWorkerV8 X: ********");
    218 }
    219 
    220 extern void WorkerV8ObjectTemplateInit(v8::Handle<v8::ObjectTemplate> target) {
    221     DBG("WorkerV8ObjectTemplateInit(target) E:");
    222     target->Set(v8::String::New("Worker"), WorkerV8Template);
    223     DBG("WorkerV8ObjectTemplateInit(target) X:\n");
    224 }
    225