1 /** Class IoTRMICall provides methods that the upper
2 * layers can use to transport and invoke methods
3 * when using IoTSocket, IoTSocketClient and IoTSocketServer.
5 * This class serves in the stub part of the RMI
6 * communication. It bridges and creates RMI requests to be
7 * transferred into the RMI object.
9 * @author Rahmadi Trimananda <rtrimana @ uci.edu>
13 #ifndef _IOTRMICALL_HPP__
14 #define _IOTRMICALL_HPP__
18 #include "IoTRMIUtil.hpp"
19 #include "IoTSocketClient.hpp"
25 IoTRMICall(int _port, const char* _address, int _rev, bool* _bResult);
28 int methodLength(string paramCls[], void* paramObj[], int numParam);
29 char* methodToBytes(string methodSign, string paramCls[], void* paramObj[],
30 char* method, int numParam);
31 void* remoteCall(string methodSign, string retType, string paramCls[],
32 void* paramObj[], int numParam, void* retObj);
36 IoTSocketClient *rmiClient;
41 IoTRMICall::IoTRMICall(int _port, const char* _address, int _rev, bool* _bResult) {
43 rmiUtil = new IoTRMIUtil();
44 if (rmiUtil == NULL) {
45 perror("IoTRMICall: IoTRMIUtil isn't initialized!");
47 rmiClient = new IoTSocketClient(_port, _address, _rev, _bResult);
48 if (rmiClient == NULL) {
49 perror("IoTRMICall: IoTSocketClient isn't initialized!");
55 IoTRMICall::~IoTRMICall() {
58 if (rmiUtil != NULL) {
63 if (rmiClient != NULL) {
73 // Calls a method remotely by passing in parameters and getting a return object
74 void* IoTRMICall::remoteCall(string methodSign, string retType, string paramCls[],
75 void* paramObj[], int numParam, void* retObj) {
77 // Send input parameters
78 int len = methodLength(paramCls, paramObj, numParam);
80 methodToBytes(methodSign, paramCls, paramObj, method, numParam);
83 rmiClient->sendBytes(method, len);
85 // Receive return value and return it to caller
86 if (retType.compare("void") == 0)
87 // Just make it NULL if it's a void return
91 char* retObjBytes = NULL;
92 retObjBytes = rmiClient->receiveBytes(retObjBytes, &retLen);
93 IoTRMIUtil::printBytes(retObjBytes, retLen, false);
94 retObj = IoTRMIUtil::getParamObject(retObj, retType.c_str(), retObjBytes, retLen);
101 // Find the bytes length of a method
102 int IoTRMICall::methodLength(string paramCls[], void* paramObj[], int numParam) {
104 // Get byte arrays and calculate method bytes length
105 // Start from the method Id...
106 int methodLen = IoTRMIUtil::METHOD_ID_LEN;
107 for (int i = 0; i < numParam; i++) {
108 // Find the parameter length
109 int paramLen = rmiUtil->getTypeSize(paramCls[i]);
110 if (paramLen == -1) { // Store the length of the field - indefinite length
111 paramLen = rmiUtil->getVarTypeSize(paramCls[i], paramObj[i]);
112 // Some space for param length, i.e. 32 bits for integer
113 methodLen = methodLen + IoTRMIUtil::PARAM_LEN;
115 // Calculate methodLen
116 methodLen = methodLen + paramLen;
123 // Convert method and its parameters into bytes
124 char* IoTRMICall::methodToBytes(string methodSign, string paramCls[],
125 void* paramObj[], char* method, int numParam) {
127 // Get method ID in bytes
128 char methodId[IoTRMIUtil::METHOD_ID_LEN];
129 IoTRMIUtil::getHashCodeBytes(methodSign, methodId);
130 memcpy(method, methodId, IoTRMIUtil::METHOD_ID_LEN);
131 int pos = IoTRMIUtil::METHOD_ID_LEN;
132 // Get byte arrays and calculate method bytes length
133 for (int i = 0; i < numParam; i++) {
134 // Find the parameter length
135 int paramLen = rmiUtil->getTypeSize(paramCls[i]);
136 if (paramLen == -1) { // Store the length of the field - indefinite length
137 paramLen = rmiUtil->getVarTypeSize(paramCls[i], paramObj[i]);
138 // Write the parameter length
139 char prmLenBytes[IoTRMIUtil::METHOD_ID_LEN];
140 IoTRMIUtil::intToByteArray(paramLen, prmLenBytes);
141 memcpy(method + pos, prmLenBytes, IoTRMIUtil::PARAM_LEN);
142 pos = pos + IoTRMIUtil::PARAM_LEN;
144 // Get array of bytes and put it in the array of array of bytes
145 char objBytes[paramLen];
146 IoTRMIUtil::getObjectBytes(objBytes, paramObj[i], paramCls[i].c_str());
147 memcpy(method + pos, objBytes, paramLen);
148 pos = pos + paramLen;