• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 <v8.h>
18 #include <telephony/ril.h>
19 
20 #include "logging.h"
21 #include "status.h"
22 #include "worker.h"
23 #include "util.h"
24 
25 #include "ril.pb.h"
26 
27 #include "logging.h"
28 #include "js_support.h"
29 #include "node_buffer.h"
30 #include "node_util.h"
31 #include "protobuf_v8.h"
32 #include "requests.h"
33 
34 #include "experiments.h"
35 
testStlPort()36 void testStlPort() {
37     // Test using STLport
38     std::queue<int *> q;
39     int data[] = {1, 2, 3};
40 
41     int *param = data;
42     LOGD("before push q.size=%d", q.size());
43     q.push(param);
44     LOGD("after push q.size=%d", q.size());
45     void *p = q.front();
46     if (p == param) {
47         LOGD("q.push succeeded");
48     } else {
49         LOGD("q.push failed");
50     }
51     q.pop();
52     LOGD("after pop q.size=%d", q.size());
53 }
54 
GetReqScreenState(v8::Local<v8::String> property,const v8::AccessorInfo & info)55 v8::Handle<v8::Value> GetReqScreenState(v8::Local<v8::String> property,
56                                const v8::AccessorInfo &info) {
57     v8::Local<v8::Object> self = info.Holder();
58     v8::Local<v8::External> wrap =
59             v8::Local<v8::External>::Cast(self->GetInternalField(0));
60     void *p = wrap->Value();
61     int state = static_cast<int *>(p)[0];
62     LOGD("GetReqScreenState state=%d", state);
63     return v8::Integer::New(state);
64 }
65 
callOnRilRequest(v8::Handle<v8::Context> context,int request,void * data,size_t datalen,RIL_Token t)66 bool callOnRilRequest(v8::Handle<v8::Context> context, int request,
67                    void *data, size_t datalen, RIL_Token t) {
68     v8::HandleScope handle_scope;
69     v8::TryCatch try_catch;
70 
71     // Get the onRilRequestFunction, making sure its a function
72     v8::Handle<v8::String> name = v8::String::New("onRilRequest");
73     v8::Handle<v8::Value> onRilRequestFunctionValue = context->Global()->Get(name);
74     if(!onRilRequestFunctionValue->IsFunction()) {
75         // Wasn't a function
76         LOGD("callOnRilRequest X wasn't a function");
77         return false;
78     }
79     v8::Handle<v8::Function> onRilRequestFunction =
80         v8::Handle<v8::Function>::Cast(onRilRequestFunctionValue);
81 
82     // Create the request
83     v8::Handle<v8::Value> v8RequestValue = v8::Number::New(request);
84 
85     // Create the parameter for the request
86     v8::Handle<v8::Object> params_obj =
87             v8::ObjectTemplate::New()->NewInstance();
88     switch(request) {
89         case(RIL_REQUEST_SCREEN_STATE): {
90             LOGD("callOnRilRequest RIL_REQUEST_SCREEN_STATE");
91             if (datalen < sizeof(int)) {
92                 LOGD("callOnRilRequest err size < sizeof int");
93             } else {
94                 v8::Handle<v8::ObjectTemplate> params_obj_template =
95                         v8::ObjectTemplate::New();
96                 params_obj_template->SetInternalFieldCount(1);
97                 params_obj_template->SetAccessor(v8::String::New(
98                             "ReqScreenState"), GetReqScreenState, NULL);
99                 // How to not leak this pointer!!!
100                 int *p = new int;
101                 *p = ((int *)data)[0];
102                 params_obj = params_obj_template->NewInstance();
103                 params_obj->SetInternalField(0, v8::External::New(p));
104             }
105             break;
106         }
107         default: {
108             LOGD("callOnRilRequest X unknown request");
109             break;
110         }
111     }
112 
113     // Invoke onRilRequest
114     bool retValue;
115     const int argc = 2;
116     v8::Handle<v8::Value> argv[argc] = { v8RequestValue, params_obj };
117     v8::Handle<v8::Value> result =
118         onRilRequestFunction->Call(context->Global(), argc, argv);
119     if (try_catch.HasCaught()) {
120         LOGD("callOnRilRequest error");
121         ReportException(&try_catch);
122         retValue = false;
123     } else {
124         v8::String::Utf8Value result_string(result);
125         LOGD("callOnRilRequest result=%s", ToCString(result_string));
126         retValue = true;
127     }
128     return retValue;
129 }
130 
testOnRilRequestUsingCppRequestObjs(v8::Handle<v8::Context> context)131 void testOnRilRequestUsingCppRequestObjs(v8::Handle<v8::Context> context) {
132     LOGD("testOnRilRequestUsingCppRequestObjs E:");
133     v8::HandleScope handle_scope;
134 
135     v8::TryCatch try_catch;
136     try_catch.SetVerbose(true);
137 
138     runJs(context, &try_catch, "local-string",
139         "function onRilRequest(reqNum, params) {\n"
140         "  print(\"reqNum=\" + reqNum);\n"
141         "  if (reqNum == 61) {\n"
142         "      print(\"params.ReqScreenState=\" + params.ReqScreenState);\n"
143         "  }\n"
144         "  return \"Hello World\";\n"
145         "}\n");
146     if (!try_catch.HasCaught()) {
147         // Call the onRilRequest function
148         int data[1] = { 0 };
149         callOnRilRequest(context, RIL_REQUEST_SCREEN_STATE, data,
150                 sizeof(data), NULL);
151     }
152     LOGD("testOnRilRequestUsingCppRequestObjs X:");
153 }
154 
testReqScreenStateProtobuf()155 void testReqScreenStateProtobuf() {
156     v8::HandleScope handle_scope;
157     v8::TryCatch try_catch;
158 
159     LOGD("testReqScreenStateProtobuf E");
160 
161     LOGD("create ReqScreenState");
162     ril_proto::ReqScreenState* ss = new ril_proto::ReqScreenState();
163     ss->set_state(true);
164     bool state = ss->state();
165     LOGD("state=%d", state);
166     ss->set_state(false);
167     state = ss->state();
168     LOGD("state=%d", state);
169     int len = ss->ByteSize();
170     LOGD("create buffer len=%d", len);
171     char *buffer = new char[len];
172     LOGD("serialize");
173     bool ok = ss->SerializeToArray(buffer, len);
174     if (!ok) {
175         LOGD("testReqScreenStateProtobuf X: Could not serialize ss");
176         return;
177     }
178     LOGD("ReqScreenState serialized ok");
179     ril_proto::ReqScreenState *newSs = new ril_proto::ReqScreenState();
180     ok = newSs->ParseFromArray(buffer, len);
181     if (!ok) {
182         LOGD("testReqScreenStateProtobuf X: Could not deserialize ss");
183         return;
184     }
185     LOGD("newSs->state=%d", newSs->state());
186 
187     delete [] buffer;
188     delete ss;
189     delete newSs;
190     LOGD("testReqScreenStateProtobuf X");
191 }
192 
testReqHangUpProtobuf()193 void testReqHangUpProtobuf() {
194     v8::HandleScope handle_scope;
195     v8::TryCatch try_catch;
196 
197     LOGD("testReqHangUpProtobuf E");
198 
199     LOGD("create ReqHangUp");
200     ril_proto::ReqHangUp* hu = new ril_proto::ReqHangUp();
201     hu->set_connection_index(3);
202     bool connection_index = hu->connection_index();
203     LOGD("connection_index=%d", connection_index);
204     hu->set_connection_index(2);
205     connection_index = hu->connection_index();
206     LOGD("connection_index=%d", connection_index);
207     LOGD("create buffer");
208     int len = hu->ByteSize();
209     char *buffer = new char[len];
210     LOGD("serialize");
211     bool ok = hu->SerializeToArray(buffer, len);
212     if (!ok) {
213         LOGD("testReqHangUpProtobuf X: Could not serialize hu");
214         return;
215     }
216     LOGD("ReqHangUp serialized ok");
217     ril_proto::ReqHangUp *newHu = new ril_proto::ReqHangUp();
218     ok = newHu->ParseFromArray(buffer, len);
219     if (!ok) {
220         LOGD("testReqHangUpProtobuf X: Could not deserialize hu");
221         return;
222     }
223     LOGD("newHu->connection_index=%d", newHu->connection_index());
224 
225     delete [] buffer;
226     delete hu;
227     delete newHu;
228     LOGD("testReqHangUpProtobuf X");
229 }
230 
testProtobufV8(v8::Handle<v8::Context> context)231 void testProtobufV8(v8::Handle<v8::Context> context) {
232     LOGD("testProtobufV8 E:");
233     v8::HandleScope handle_scope;
234 
235     v8::TryCatch try_catch;
236     try_catch.SetVerbose(true);
237 
238     if (try_catch.HasCaught()) {
239         LOGD("TryCatch.hasCaught is true after protobuf_v8::init");
240         ReportException(&try_catch);
241     }
242     runJs(context, &try_catch, "local-string",
243         "fileContents = readFileToString('mock_ril.js');\n"
244         "print('fileContents:\\n' + fileContents);\n"
245         "\n"
246         "buffer = readFileToBuffer('ril.desc');\n"
247         "var schema = new Schema(buffer);\n"
248         "\n"
249         "var originalReqEnterSimPin = { pin : 'hello-the-pin' };\n"
250         "print('originalReqEnterSimPin: pin=' + originalReqEnterSimPin.pin);\n"
251         "var ReqEnterSimPinSchema = schema['ril_proto.ReqEnterSimPin'];\n"
252         "serializedOriginalReqEnterSimPin = ReqEnterSimPinSchema.serialize(originalReqEnterSimPin);\n"
253         "print('serializedOriginalReqEnterSimPin.length=' + serializedOriginalReqEnterSimPin.length);\n"
254         "newReqEnterSimPin = ReqEnterSimPinSchema.parse(serializedOriginalReqEnterSimPin);\n"
255         "print('newReqEnterSimPin: pin=' + newReqEnterSimPin.pin);\n"
256         "\n"
257         "var originalReqScreenState = { state : true };\n"
258         "print('originalReqScreenState: state=' + originalReqScreenState.state);\n"
259         "var ReqScreenStateSchema = schema['ril_proto.ReqScreenState'];\n"
260         "var serializedOriginalReqScreenState = ReqScreenStateSchema.serialize(originalReqScreenState);\n"
261         "print('serializedOriginalReqScreenState.length=' + serializedOriginalReqScreenState.length);\n"
262         "var newReqScreenState = ReqScreenStateSchema.parse(serializedOriginalReqScreenState);\n"
263         "print('newReqScreenState: state=' + newReqScreenState.state);\n"
264         "\n"
265         "originalReqScreenState.state = false;\n"
266         "print('originalReqScreenState: state=' + originalReqScreenState.state);\n"
267         "serializedOriginalReqScreenState = ReqScreenStateSchema.serialize(originalReqScreenState);\n"
268         "print('serializedOriginalReqScreenState.length=' + serializedOriginalReqScreenState.length);\n"
269         "newReqScreenState = ReqScreenStateSchema.parse(serializedOriginalReqScreenState);\n"
270         "print('newReqScreenState: state=' + newReqScreenState.state);\n");
271     LOGD("testProtobufV8 X");
272 }
273 
experiments(v8::Handle<v8::Context> context)274 void experiments(v8::Handle<v8::Context> context) {
275     LOGD("experiments E: ********");
276     testStlPort();
277     testReqScreenStateProtobuf();
278     testOnRilRequestUsingCppRequestObjs(context);
279     testProtobufV8(context);
280     LOGD("experiments X: ********\n");
281 }
282