/* ***** 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 **) calloc(count + 1, sizeof(char *)); if (!strs) return NULL; count = 0; for (node = nodes; node; node = node->mNext, ++count) strs[count] = (char *) node->Value(); 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 ipcmMessagePing()); } static void OnClientHello(ipcClient *client, const ipcMessage *rawMsg) { LOG(("got CLIENT_HELLO\n")); IPC_SendMsg(client, new ipcmMessageClientID(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); } 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); } static void OnClientAddTarget(ipcClient *client, const ipcMessage *rawMsg) { LOG(("got CLIENT_ADD_TARGET\n")); ipcMessageCast msg(rawMsg); client->AddTarget(msg->Target()); } static void OnClientDelTarget(ipcClient *client, const ipcMessage *rawMsg) { LOG(("got CLIENT_DEL_TARGET\n")); ipcMessageCast msg(rawMsg); client->DelTarget(msg->Target()); } static void OnQueryClientByName(ipcClient *client, const ipcMessage *rawMsg) { LOG(("got QUERY_CLIENT_BY_NAME\n")); 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(result->ID())); } else { LOG((" client does not exist\n")); IPC_SendMsg(client, new ipcmMessageError(IPCM_ERROR_CLIENT_NOT_FOUND)); } } 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(), (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)); } } static void OnForward(ipcClient *client, const ipcMessage *rawMsg) { LOG(("got FORWARD\n")); ipcMessageCast msg(rawMsg); ipcClient *dest = IPC_GetClientByID(msg->DestClientID()); if (!dest) { LOG((" destination client not found!\n")); return; } ipcMessage *newMsg = new ipcMessage(msg->InnerTarget(), msg->InnerData(), msg->InnerDataLen()); IPC_SendMsg(dest, newMsg); } }; void IPCM_HandleMsg(ipcClient *client, const ipcMessage *rawMsg) { static ipcCommandModule::MsgHandler handlers[] = { ipcCommandModule::OnPing, NULL, // ERROR ipcCommandModule::OnClientHello, NULL, // CLIENT_ID NULL, // CLIENT_INFO ipcCommandModule::OnClientAddName, ipcCommandModule::OnClientDelName, ipcCommandModule::OnClientAddTarget, ipcCommandModule::OnClientDelTarget, ipcCommandModule::OnQueryClientByName, ipcCommandModule::OnQueryClientInfo, ipcCommandModule::OnForward, }; int type = IPCM_GetMsgType(rawMsg); LOG(("IPCM_HandleMsg [type=%d]\n", type)); if (type < IPCM_MSG_TYPE_UNKNOWN) { if (handlers[type]) { ipcCommandModule::MsgHandler handler = handlers[type]; handler(client, rawMsg); } } }