/* ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Mozilla Public License Version * 1.1 (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * http://www.mozilla.org/MPL/ * * Software distributed under the License is distributed on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License * for the specific language governing rights and limitations under the * License. * * The Original Code is Mozilla IPC. * * The Initial Developer of the Original Code is * Netscape Communications Corporation. * Portions created by the Initial Developer are Copyright (C) 2002 * the Initial Developer. All Rights Reserved. * * Contributor(s): * Darin Fisher * * Alternatively, the contents of this file may be used under the terms of * either the GNU General Public License Version 2 or later (the "GPL"), or * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ #include #include #include "ipcLog.h" #include "ipcCommandModule.h" #include "ipcModule.h" #include "ipcClient.h" #include "ipcMessage.h" #include "ipcMessageUtils.h" #include "ipcModuleReg.h" #include "ipcd.h" #include "ipcm.h" struct ipcCommandModule { typedef void (* MsgHandler)(ipcClient *, const ipcMessage *); // // helpers // static char ** BuildStringArray(const ipcStringNode *nodes) { size_t count = 0; const ipcStringNode *node; for (node = nodes; node; node = node->mNext) count++; char **strs = (char **) malloc((count + 1) * sizeof(char *)); if (!strs) return NULL; count = 0; for (node = nodes; node; node = node->mNext, ++count) strs[count] = (char *) node->Value(); strs[count] = 0; return strs; } static nsID ** BuildIDArray(const ipcIDNode *nodes) { size_t count = 0; const ipcIDNode *node; for (node = nodes; node; node = node->mNext) count++; nsID **ids = (nsID **) calloc(count + 1, sizeof(nsID *)); if (!ids) return NULL; count = 0; for (node = nodes; node; node = node->mNext, ++count) ids[count] = (nsID *) &node->Value(); return ids; } // // message handlers // static void OnPing(ipcClient *client, const ipcMessage *rawMsg) { LOG(("got PING\n")); IPC_SendMsg(client, new ipcmMessageResult(IPCM_GetRequestIndex(rawMsg), IPCM_OK)); } static void OnClientHello(ipcClient *client, const ipcMessage *rawMsg) { LOG(("got CLIENT_HELLO\n")); IPC_SendMsg(client, new ipcmMessageClientID(IPCM_GetRequestIndex(rawMsg), client->ID())); // // NOTE: it would almost make sense for this notification to live // in the transport layer code. however, clients expect to receive // a CLIENT_ID as the first message following a CLIENT_HELLO, so we // must not allow modules to see a client until after we have sent // the CLIENT_ID message. // IPC_NotifyClientUp(client); } static void OnClientAddName(ipcClient *client, const ipcMessage *rawMsg) { LOG(("got CLIENT_ADD_NAME\n")); ipcMessageCast msg(rawMsg); const char *name = msg->Name(); if (name) client->AddName(name); // TODO: send better status code (e.g., suppose name already defined for client) IPC_SendMsg(client, new ipcmMessageResult(IPCM_GetRequestIndex(rawMsg), IPCM_OK)); } static void OnClientDelName(ipcClient *client, const ipcMessage *rawMsg) { LOG(("got CLIENT_DEL_NAME\n")); ipcMessageCast msg(rawMsg); const char *name = msg->Name(); if (name) client->DelName(name); // TODO: send better status code (e.g., suppose name not defined for client) IPC_SendMsg(client, new ipcmMessageResult(IPCM_GetRequestIndex(rawMsg), IPCM_OK)); } static void OnClientAddTarget(ipcClient *client, const ipcMessage *rawMsg) { LOG(("got CLIENT_ADD_TARGET\n")); ipcMessageCast msg(rawMsg); client->AddTarget(msg->Target()); // TODO: send better status code (e.g., suppose target already defined for client) IPC_SendMsg(client, new ipcmMessageResult(IPCM_GetRequestIndex(rawMsg), IPCM_OK)); } static void OnClientDelTarget(ipcClient *client, const ipcMessage *rawMsg) { LOG(("got CLIENT_DEL_TARGET\n")); ipcMessageCast msg(rawMsg); client->DelTarget(msg->Target()); // TODO: send better status code (e.g., suppose target not defined for client) IPC_SendMsg(client, new ipcmMessageResult(IPCM_GetRequestIndex(rawMsg), IPCM_OK)); } static void OnQueryClientByName(ipcClient *client, const ipcMessage *rawMsg) { LOG(("got QUERY_CLIENT_BY_NAME\n")); PRUint32 requestIndex = IPCM_GetRequestIndex(rawMsg); ipcMessageCast msg(rawMsg); ipcClient *result = IPC_GetClientByName(msg->Name()); if (result) { LOG((" client exists w/ ID = %u\n", result->ID())); IPC_SendMsg(client, new ipcmMessageClientID(requestIndex, result->ID())); } else { LOG((" client does not exist\n")); IPC_SendMsg(client, new ipcmMessageResult(requestIndex, IPCM_ERROR_NO_CLIENT)); } } #if 0 static void OnQueryClientInfo(ipcClient *client, const ipcMessage *rawMsg) { LOG(("got QUERY_CLIENT_INFO\n")); ipcMessageCast msg(rawMsg); ipcClient *result = IPC_GetClientByID(msg->ClientID()); if (result) { char **names = BuildStringArray(result->Names()); nsID **targets = BuildIDArray(result->Targets()); IPC_SendMsg(client, new ipcmMessageClientInfo(result->ID(), msg->RequestIndex(), (const char **) names, (const nsID **) targets)); free(names); free(targets); } else { LOG((" client does not exist\n")); IPC_SendMsg(client, new ipcmMessageError(IPCM_ERROR_CLIENT_NOT_FOUND, msg->RequestIndex())); } } #endif static void OnForward(ipcClient *client, const ipcMessage *rawMsg) { LOG(("got FORWARD\n")); ipcMessageCast msg(rawMsg); ipcClient *dest = IPC_GetClientByID(msg->ClientID()); if (!dest) { LOG((" destination client not found!\n")); IPC_SendMsg(client, new ipcmMessageResult(IPCM_GetRequestIndex(rawMsg), IPCM_ERROR_NO_CLIENT)); return; } // inform client that its message will be forwarded IPC_SendMsg(client, new ipcmMessageResult(IPCM_GetRequestIndex(rawMsg), IPCM_OK)); ipcMessage *newMsg = new ipcmMessageForward(IPCM_MSG_PSH_FORWARD, client->ID(), msg->InnerTarget(), msg->InnerData(), msg->InnerDataLen()); IPC_SendMsg(dest, newMsg); } }; void IPCM_HandleMsg(ipcClient *client, const ipcMessage *rawMsg) { static ipcCommandModule::MsgHandler handlers[] = { ipcCommandModule::OnPing, ipcCommandModule::OnForward, ipcCommandModule::OnClientHello, ipcCommandModule::OnClientAddName, ipcCommandModule::OnClientDelName, ipcCommandModule::OnClientAddTarget, ipcCommandModule::OnClientDelTarget, ipcCommandModule::OnQueryClientByName }; int type = IPCM_GetType(rawMsg); LOG(("IPCM_HandleMsg [type=%x]\n", type)); if (!(type & IPCM_MSG_CLASS_REQ)) { LOG(("not a request -- ignoring message\n")); return; } type &= ~IPCM_MSG_CLASS_REQ; type--; if (type < 0 || type >= (int) (sizeof(handlers)/sizeof(handlers[0]))) { LOG(("unknown request -- ignoring message\n")); return; } (handlers[type])(client, rawMsg); }