624 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			624 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*******************************************************************************
 | |
|  *
 | |
|  * Copyright (c) 2000-2003 Intel Corporation 
 | |
|  * All rights reserved. 
 | |
|  *
 | |
|  * Redistribution and use in source and binary forms, with or without 
 | |
|  * modification, are permitted provided that the following conditions are met: 
 | |
|  *
 | |
|  * - Redistributions of source code must retain the above copyright notice, 
 | |
|  * this list of conditions and the following disclaimer. 
 | |
|  * - Redistributions in binary form must reproduce the above copyright notice, 
 | |
|  * this list of conditions and the following disclaimer in the documentation 
 | |
|  * and/or other materials provided with the distribution. 
 | |
|  * - Neither name of Intel Corporation nor the names of its contributors 
 | |
|  * may be used to endorse or promote products derived from this software 
 | |
|  * without specific prior written permission.
 | |
|  * 
 | |
|  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 
 | |
|  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 
 | |
|  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 
 | |
|  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL INTEL OR 
 | |
|  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, 
 | |
|  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, 
 | |
|  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 
 | |
|  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY 
 | |
|  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
 | |
|  * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS 
 | |
|  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 | |
|  *
 | |
|  ******************************************************************************/
 | |
| 
 | |
| 
 | |
| #include "sample_util.h"
 | |
| 
 | |
| 
 | |
| #include <assert.h>
 | |
| #include <stdarg.h>
 | |
| #include <stdio.h>
 | |
| 
 | |
| 
 | |
| #if !UPNP_HAVE_TOOLS
 | |
| #	error "Need upnptools.h to compile samples ; try ./configure --enable-tools"
 | |
| #endif
 | |
| 
 | |
| 
 | |
| static int initialize_init = 1;
 | |
| static int initialize_register = 1;
 | |
| 
 | |
| /*! Function pointers to use for displaying formatted strings.
 | |
|  * Set on Initialization of device. */
 | |
| print_string gPrintFun = NULL;
 | |
| state_update gStateUpdateFun = NULL;
 | |
| 
 | |
| /*! mutex to control displaying of events */
 | |
| ithread_mutex_t display_mutex;
 | |
| 
 | |
| int SampleUtil_Initialize(print_string print_function)
 | |
| {
 | |
| 	if (initialize_init) {
 | |
| 		ithread_mutexattr_t attr;
 | |
| 
 | |
| 		ithread_mutexattr_init(&attr);
 | |
| 		ithread_mutexattr_setkind_np(&attr, ITHREAD_MUTEX_RECURSIVE_NP);
 | |
| 		ithread_mutex_init(&display_mutex, &attr);
 | |
| 		ithread_mutexattr_destroy(&attr);
 | |
| 
 | |
| 		/* To shut up valgrind mutex warning. */
 | |
| 		ithread_mutex_lock(&display_mutex);
 | |
| 		gPrintFun = print_function;
 | |
| 		ithread_mutex_unlock(&display_mutex);
 | |
| 
 | |
| 		initialize_init = 0;
 | |
| 	}
 | |
| 
 | |
| 	return UPNP_E_SUCCESS;
 | |
| }
 | |
| 
 | |
| int SampleUtil_RegisterUpdateFunction(state_update update_function)
 | |
| {
 | |
| 	if (initialize_register) {
 | |
| 		gStateUpdateFun = update_function;
 | |
| 		initialize_register = 0;
 | |
| 	}
 | |
| 
 | |
| 	return UPNP_E_SUCCESS;
 | |
| }
 | |
| 
 | |
| int SampleUtil_Finish()
 | |
| {
 | |
| 	ithread_mutex_destroy(&display_mutex);
 | |
| 	gPrintFun = NULL;
 | |
| 	gStateUpdateFun = NULL;
 | |
| 	initialize_init = 1;
 | |
| 	initialize_register = 1;
 | |
| 
 | |
| 	return UPNP_E_SUCCESS;
 | |
| }
 | |
| 
 | |
| char *SampleUtil_GetElementValue(IXML_Element *element)
 | |
| {
 | |
| 	IXML_Node *child = ixmlNode_getFirstChild((IXML_Node *)element);
 | |
| 	char *temp = NULL;
 | |
| 
 | |
| 	if (child != 0 && ixmlNode_getNodeType(child) == eTEXT_NODE) {
 | |
| 		temp = strdup(ixmlNode_getNodeValue(child));
 | |
| 	}
 | |
| 
 | |
| 	return temp;
 | |
| }
 | |
| 
 | |
| IXML_NodeList *SampleUtil_GetFirstServiceList(IXML_Document *doc)
 | |
| {
 | |
| 	IXML_NodeList *ServiceList = NULL;
 | |
| 	IXML_NodeList *servlistnodelist = NULL;
 | |
| 	IXML_Node *servlistnode = NULL;
 | |
| 
 | |
| 	servlistnodelist =
 | |
| 		ixmlDocument_getElementsByTagName( doc, "serviceList" );
 | |
| 	if (servlistnodelist && ixmlNodeList_length(servlistnodelist)) {
 | |
| 		/* we only care about the first service list, from the root
 | |
| 		 * device */
 | |
| 		servlistnode = ixmlNodeList_item(servlistnodelist, 0);
 | |
| 		/* create as list of DOM nodes */
 | |
| 		ServiceList = ixmlElement_getElementsByTagName(
 | |
| 			(IXML_Element *)servlistnode, "service");
 | |
| 	}
 | |
| 	if (servlistnodelist) {
 | |
| 		ixmlNodeList_free(servlistnodelist);
 | |
| 	}
 | |
| 
 | |
| 	return ServiceList;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Obtain the service list 
 | |
|  *    n == 0 the first
 | |
|  *    n == 1 the next in the device list, etc..
 | |
|  */
 | |
| static IXML_NodeList *SampleUtil_GetNthServiceList(
 | |
| 	/*! [in] . */
 | |
| 	IXML_Document *doc,
 | |
| 	/*! [in] . */
 | |
| 	int n)
 | |
| {
 | |
| 	IXML_NodeList *ServiceList = NULL;
 | |
| 	IXML_NodeList *servlistnodelist = NULL;
 | |
| 	IXML_Node *servlistnode = NULL;
 | |
| 
 | |
| 	/*  ixmlDocument_getElementsByTagName()
 | |
| 	 *  Returns a NodeList of all Elements that match the given
 | |
| 	 *  tag name in the order in which they were encountered in a preorder
 | |
| 	 *  traversal of the Document tree.  
 | |
| 	 *
 | |
| 	 *  return (NodeList*) A pointer to a NodeList containing the 
 | |
| 	 *                      matching items or NULL on an error. 	 */
 | |
| 	SampleUtil_Print("SampleUtil_GetNthServiceList called : n = %d\n", n);
 | |
| 	servlistnodelist =
 | |
| 		ixmlDocument_getElementsByTagName(doc, "serviceList");
 | |
| 	if (servlistnodelist &&
 | |
| 	    ixmlNodeList_length(servlistnodelist) &&
 | |
| 	    n < ixmlNodeList_length(servlistnodelist)) {
 | |
| 		/* For the first service list (from the root device),
 | |
| 		 * we pass 0 */
 | |
| 		/*servlistnode = ixmlNodeList_item( servlistnodelist, 0 );*/
 | |
| 
 | |
| 		/* Retrieves a Node from a NodeList} specified by a 
 | |
| 		 *  numerical index.
 | |
| 		 *
 | |
| 		 *  return (Node*) A pointer to a Node or NULL if there was an 
 | |
| 		 *                  error. */
 | |
| 		servlistnode = ixmlNodeList_item(servlistnodelist, n);
 | |
| 
 | |
| 		assert(servlistnode != 0);
 | |
| 
 | |
| 		/* create as list of DOM nodes */
 | |
| 		ServiceList = ixmlElement_getElementsByTagName(
 | |
| 			(IXML_Element *)servlistnode, "service");
 | |
| 	}
 | |
| 
 | |
| 	if (servlistnodelist) {
 | |
| 		ixmlNodeList_free(servlistnodelist);
 | |
| 	}
 | |
| 
 | |
| 	return ServiceList;
 | |
| }
 | |
| 
 | |
| char *SampleUtil_GetFirstDocumentItem(IXML_Document *doc, const char *item)
 | |
| {
 | |
| 	IXML_NodeList *nodeList = NULL;
 | |
| 	IXML_Node *textNode = NULL;
 | |
| 	IXML_Node *tmpNode = NULL;
 | |
| 	char *ret = NULL;
 | |
| 
 | |
| 	nodeList = ixmlDocument_getElementsByTagName(doc, (char *)item);
 | |
| 	if (nodeList) {
 | |
| 		tmpNode = ixmlNodeList_item(nodeList, 0);
 | |
| 		if (tmpNode) {
 | |
| 			textNode = ixmlNode_getFirstChild(tmpNode);
 | |
| 			if (!textNode) {
 | |
| 				SampleUtil_Print("sample_util.c: (bug) "
 | |
| 					"ixmlNode_getFirstChild(tmpNode) "
 | |
| 					"returned NULL\n"); 
 | |
| 				ret = strdup("");
 | |
| 				goto epilogue;
 | |
| 			}
 | |
| 			if (!ixmlNode_getNodeValue(textNode)) {
 | |
| 				SampleUtil_Print("ixmlNode_getNodeValue "
 | |
| 					"returned NULL\n"); 
 | |
| 				ret = strdup("");
 | |
| 				goto epilogue;
 | |
| 			} else {
 | |
| 				ret = strdup(ixmlNode_getNodeValue(textNode));
 | |
| 			}
 | |
| 		} else {
 | |
| 			SampleUtil_Print("ixmlNode_getFirstChild(tmpNode) "
 | |
| 				"returned NULL\n");
 | |
| 			goto epilogue;
 | |
| 		}
 | |
| 	} else {
 | |
| 		SampleUtil_Print("Error finding %s in XML Node\n", item);
 | |
| 		goto epilogue;
 | |
| 	}
 | |
| 
 | |
| epilogue:
 | |
| 	if (nodeList) {
 | |
| 		ixmlNodeList_free(nodeList);
 | |
| 	}
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| char *SampleUtil_GetFirstElementItem(IXML_Element *element, const char *item)
 | |
| {
 | |
| 	IXML_NodeList *nodeList = NULL;
 | |
| 	IXML_Node *textNode = NULL;
 | |
| 	IXML_Node *tmpNode = NULL;
 | |
| 	char *ret = NULL;
 | |
| 
 | |
| 	nodeList = ixmlElement_getElementsByTagName(element, (char *)item);
 | |
| 	if (nodeList == NULL) {
 | |
| 		SampleUtil_Print( "Error finding %s in XML Node\n", item);
 | |
| 
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	tmpNode = ixmlNodeList_item(nodeList, 0);
 | |
| 	if (tmpNode) {
 | |
| 		SampleUtil_Print("Error finding %s value in XML Node\n", item);
 | |
| 		ixmlNodeList_free(nodeList);
 | |
| 
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	textNode = ixmlNode_getFirstChild(tmpNode);
 | |
| 	ret = strdup(ixmlNode_getNodeValue(textNode));
 | |
| 	if (!ret) {
 | |
| 		SampleUtil_Print("Error allocating memory for %s in XML Node\n",
 | |
| 			item);
 | |
| 		ixmlNodeList_free(nodeList);
 | |
| 
 | |
| 		return NULL;
 | |
| 	}
 | |
| 	ixmlNodeList_free(nodeList);
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| void SampleUtil_PrintEventType(Upnp_EventType S)
 | |
| {
 | |
| 	switch (S) {
 | |
| 	/* Discovery */
 | |
| 	case UPNP_DISCOVERY_ADVERTISEMENT_ALIVE:
 | |
| 		SampleUtil_Print("UPNP_DISCOVERY_ADVERTISEMENT_ALIVE\n");
 | |
| 		break;
 | |
| 	case UPNP_DISCOVERY_ADVERTISEMENT_BYEBYE:
 | |
| 		SampleUtil_Print("UPNP_DISCOVERY_ADVERTISEMENT_BYEBYE\n");
 | |
| 		break;
 | |
| 	case UPNP_DISCOVERY_SEARCH_RESULT:
 | |
| 		SampleUtil_Print( "UPNP_DISCOVERY_SEARCH_RESULT\n");
 | |
| 		break;
 | |
| 	case UPNP_DISCOVERY_SEARCH_TIMEOUT:
 | |
| 		SampleUtil_Print( "UPNP_DISCOVERY_SEARCH_TIMEOUT\n");
 | |
| 		break;
 | |
| 	/* SOAP */
 | |
| 	case UPNP_CONTROL_ACTION_REQUEST:
 | |
| 		SampleUtil_Print("UPNP_CONTROL_ACTION_REQUEST\n");
 | |
| 		break;
 | |
| 	case UPNP_CONTROL_ACTION_COMPLETE:
 | |
| 		SampleUtil_Print("UPNP_CONTROL_ACTION_COMPLETE\n");
 | |
| 		break;
 | |
| 	case UPNP_CONTROL_GET_VAR_REQUEST:
 | |
| 		SampleUtil_Print("UPNP_CONTROL_GET_VAR_REQUEST\n");
 | |
| 		break;
 | |
| 	case UPNP_CONTROL_GET_VAR_COMPLETE:
 | |
| 		SampleUtil_Print("UPNP_CONTROL_GET_VAR_COMPLETE\n");
 | |
| 		break;
 | |
| 	/* GENA */
 | |
| 	case UPNP_EVENT_SUBSCRIPTION_REQUEST:
 | |
| 		SampleUtil_Print("UPNP_EVENT_SUBSCRIPTION_REQUEST\n");
 | |
| 		break;
 | |
| 	case UPNP_EVENT_RECEIVED:
 | |
| 		SampleUtil_Print("UPNP_EVENT_RECEIVED\n");
 | |
| 		break;
 | |
| 	case UPNP_EVENT_RENEWAL_COMPLETE:
 | |
| 		SampleUtil_Print("UPNP_EVENT_RENEWAL_COMPLETE\n");
 | |
| 		break;
 | |
| 	case UPNP_EVENT_SUBSCRIBE_COMPLETE:
 | |
| 		SampleUtil_Print("UPNP_EVENT_SUBSCRIBE_COMPLETE\n");
 | |
| 		break;
 | |
| 	case UPNP_EVENT_UNSUBSCRIBE_COMPLETE:
 | |
| 		SampleUtil_Print("UPNP_EVENT_UNSUBSCRIBE_COMPLETE\n");
 | |
| 		break;
 | |
| 	case UPNP_EVENT_AUTORENEWAL_FAILED:
 | |
| 		SampleUtil_Print("UPNP_EVENT_AUTORENEWAL_FAILED\n");
 | |
| 		break;
 | |
| 	case UPNP_EVENT_SUBSCRIPTION_EXPIRED:
 | |
| 		SampleUtil_Print("UPNP_EVENT_SUBSCRIPTION_EXPIRED\n");
 | |
| 		break;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| int SampleUtil_PrintEvent(Upnp_EventType EventType, void *Event)
 | |
| {
 | |
| 	ithread_mutex_lock(&display_mutex);
 | |
| 
 | |
| 	SampleUtil_Print(
 | |
| 		"======================================================================\n"
 | |
| 		"----------------------------------------------------------------------\n");
 | |
| 	SampleUtil_PrintEventType(EventType);
 | |
| 	switch (EventType) {
 | |
| 	/* SSDP */
 | |
| 	case UPNP_DISCOVERY_ADVERTISEMENT_ALIVE:
 | |
| 	case UPNP_DISCOVERY_ADVERTISEMENT_BYEBYE:
 | |
| 	case UPNP_DISCOVERY_SEARCH_RESULT: {
 | |
| 		struct Upnp_Discovery *d_event = (struct Upnp_Discovery *)Event;
 | |
| 
 | |
| 		SampleUtil_Print("ErrCode     =  %s(%d)\n",
 | |
| 			UpnpGetErrorMessage(d_event->ErrCode), d_event->ErrCode);
 | |
| 		SampleUtil_Print("Expires     =  %d\n",  d_event->Expires);
 | |
| 		SampleUtil_Print("DeviceId    =  %s\n",  d_event->DeviceId);
 | |
| 		SampleUtil_Print("DeviceType  =  %s\n",  d_event->DeviceType);
 | |
| 		SampleUtil_Print("ServiceType =  %s\n",  d_event->ServiceType);
 | |
| 		SampleUtil_Print("ServiceVer  =  %s\n",  d_event->ServiceVer);
 | |
| 		SampleUtil_Print("Location    =  %s\n",  d_event->Location);
 | |
| 		SampleUtil_Print("OS          =  %s\n",  d_event->Os);
 | |
| 		SampleUtil_Print("Ext         =  %s\n",  d_event->Ext);
 | |
| 		break;
 | |
| 	}
 | |
| 	case UPNP_DISCOVERY_SEARCH_TIMEOUT:
 | |
| 		/* Nothing to print out here */
 | |
| 		break;
 | |
| 	/* SOAP */
 | |
| 	case UPNP_CONTROL_ACTION_REQUEST: {
 | |
| 		struct Upnp_Action_Request *a_event =
 | |
| 			(struct Upnp_Action_Request *)Event;
 | |
| 		char *xmlbuff = NULL;
 | |
| 
 | |
| 		SampleUtil_Print("ErrCode     =  %s(%d)\n",
 | |
| 			UpnpGetErrorMessage(a_event->ErrCode), a_event->ErrCode);
 | |
| 		SampleUtil_Print("ErrStr      =  %s\n", a_event->ErrStr);
 | |
| 		SampleUtil_Print("ActionName  =  %s\n", a_event->ActionName);
 | |
| 		SampleUtil_Print("UDN         =  %s\n", a_event->DevUDN);
 | |
| 		SampleUtil_Print("ServiceID   =  %s\n", a_event->ServiceID);
 | |
| 		if (a_event->ActionRequest) {
 | |
| 			xmlbuff = ixmlPrintNode((IXML_Node *)a_event->ActionRequest);
 | |
| 			if (xmlbuff) {
 | |
| 				SampleUtil_Print("ActRequest  =  %s\n", xmlbuff);
 | |
| 				ixmlFreeDOMString(xmlbuff);
 | |
| 			}
 | |
| 			xmlbuff = NULL;
 | |
| 		} else {
 | |
| 			SampleUtil_Print("ActRequest  =  (null)\n");
 | |
| 		}
 | |
| 		if (a_event->ActionResult) {
 | |
| 			xmlbuff = ixmlPrintNode((IXML_Node *)a_event->ActionResult);
 | |
| 			if (xmlbuff) {
 | |
| 				SampleUtil_Print("ActResult   =  %s\n", xmlbuff);
 | |
| 				ixmlFreeDOMString(xmlbuff);
 | |
| 			}
 | |
| 			xmlbuff = NULL;
 | |
| 		} else {
 | |
| 			SampleUtil_Print("ActResult   =  (null)\n");
 | |
| 		}
 | |
| 		break;
 | |
| 	}
 | |
| 	case UPNP_CONTROL_ACTION_COMPLETE: {
 | |
| 		struct Upnp_Action_Complete *a_event =
 | |
| 			(struct Upnp_Action_Complete *)Event;
 | |
| 		char *xmlbuff = NULL;
 | |
| 
 | |
| 		SampleUtil_Print("ErrCode     =  %s(%d)\n",  
 | |
| 			UpnpGetErrorMessage(a_event->ErrCode), a_event->ErrCode);
 | |
| 		SampleUtil_Print("CtrlUrl     =  %s\n", a_event->CtrlUrl);
 | |
| 		if (a_event->ActionRequest) {
 | |
| 			xmlbuff = ixmlPrintNode((IXML_Node *)a_event->ActionRequest);
 | |
| 			if (xmlbuff) {
 | |
| 				SampleUtil_Print("ActRequest  =  %s\n", xmlbuff);
 | |
| 				ixmlFreeDOMString(xmlbuff);
 | |
| 			}
 | |
| 			xmlbuff = NULL;
 | |
| 		} else {
 | |
| 			SampleUtil_Print("ActRequest  =  (null)\n");
 | |
| 		}
 | |
| 		if (a_event->ActionResult) {
 | |
| 			xmlbuff = ixmlPrintNode((IXML_Node *)a_event->ActionResult);
 | |
| 			if (xmlbuff) {
 | |
| 				SampleUtil_Print("ActResult   =  %s\n", xmlbuff);
 | |
| 				ixmlFreeDOMString(xmlbuff);
 | |
| 			}
 | |
| 			xmlbuff = NULL;
 | |
| 		} else {
 | |
| 			SampleUtil_Print("ActResult   =  (null)\n");
 | |
| 		}
 | |
| 		break;
 | |
| 	}
 | |
| 	case UPNP_CONTROL_GET_VAR_REQUEST: {
 | |
| 		struct Upnp_State_Var_Request *sv_event =
 | |
| 			(struct Upnp_State_Var_Request *)Event;
 | |
| 
 | |
| 		SampleUtil_Print("ErrCode     =  %s(%d)\n",
 | |
| 			UpnpGetErrorMessage(sv_event->ErrCode), sv_event->ErrCode);
 | |
| 		SampleUtil_Print("ErrStr      =  %s\n", sv_event->ErrStr);
 | |
| 		SampleUtil_Print("UDN         =  %s\n", sv_event->DevUDN);
 | |
| 		SampleUtil_Print("ServiceID   =  %s\n", sv_event->ServiceID);
 | |
| 		SampleUtil_Print("StateVarName=  %s\n", sv_event->StateVarName);
 | |
| 		SampleUtil_Print("CurrentVal  =  %s\n", sv_event->CurrentVal);
 | |
| 		break;
 | |
| 	}
 | |
| 	case UPNP_CONTROL_GET_VAR_COMPLETE: {
 | |
| 		struct Upnp_State_Var_Complete *sv_event =
 | |
| 			(struct Upnp_State_Var_Complete *)Event;
 | |
| 
 | |
| 		SampleUtil_Print("ErrCode     =  %s(%d)\n",
 | |
| 			UpnpGetErrorMessage(sv_event->ErrCode), sv_event->ErrCode);
 | |
| 		SampleUtil_Print("CtrlUrl     =  %s\n", sv_event->CtrlUrl);
 | |
| 		SampleUtil_Print("StateVarName=  %s\n", sv_event->StateVarName);
 | |
| 		SampleUtil_Print("CurrentVal  =  %s\n", sv_event->CurrentVal);
 | |
| 		break;
 | |
| 	}
 | |
| 	/* GENA */
 | |
| 	case UPNP_EVENT_SUBSCRIPTION_REQUEST: {
 | |
| 		struct Upnp_Subscription_Request *sr_event =
 | |
| 			(struct Upnp_Subscription_Request *)Event;
 | |
| 
 | |
| 		SampleUtil_Print("ServiceID   =  %s\n", sr_event->ServiceId);
 | |
| 		SampleUtil_Print("UDN         =  %s\n", sr_event->UDN);
 | |
| 		SampleUtil_Print("SID         =  %s\n", sr_event->Sid);
 | |
| 		break;
 | |
| 	}
 | |
| 	case UPNP_EVENT_RECEIVED: {
 | |
| 		struct Upnp_Event *e_event = (struct Upnp_Event *)Event;
 | |
| 		char *xmlbuff = NULL;
 | |
| 
 | |
| 		SampleUtil_Print("SID         =  %s\n", e_event->Sid);
 | |
| 		SampleUtil_Print("EventKey    =  %d\n",	e_event->EventKey);
 | |
| 		xmlbuff = ixmlPrintNode((IXML_Node *)e_event->ChangedVariables);
 | |
| 		SampleUtil_Print("ChangedVars =  %s\n", xmlbuff);
 | |
| 		ixmlFreeDOMString(xmlbuff);
 | |
| 		xmlbuff = NULL;
 | |
| 		break;
 | |
| 	}
 | |
| 	case UPNP_EVENT_RENEWAL_COMPLETE: {
 | |
| 		struct Upnp_Event_Subscribe *es_event =
 | |
| 			(struct Upnp_Event_Subscribe *)Event;
 | |
| 
 | |
| 		SampleUtil_Print("SID         =  %s\n", es_event->Sid);
 | |
| 		SampleUtil_Print("ErrCode     =  %s(%d)\n",
 | |
| 			UpnpGetErrorMessage(es_event->ErrCode), es_event->ErrCode);
 | |
| 		SampleUtil_Print("TimeOut     =  %d\n", es_event->TimeOut);
 | |
| 		break;
 | |
| 	}
 | |
| 	case UPNP_EVENT_SUBSCRIBE_COMPLETE:
 | |
| 	case UPNP_EVENT_UNSUBSCRIBE_COMPLETE: {
 | |
| 		struct Upnp_Event_Subscribe *es_event =
 | |
| 			(struct Upnp_Event_Subscribe *)Event;
 | |
| 
 | |
| 		SampleUtil_Print("SID         =  %s\n", es_event->Sid);
 | |
| 		SampleUtil_Print("ErrCode     =  %s(%d)\n",
 | |
| 			UpnpGetErrorMessage(es_event->ErrCode), es_event->ErrCode);
 | |
| 		SampleUtil_Print("PublisherURL=  %s\n", es_event->PublisherUrl);
 | |
| 		SampleUtil_Print("TimeOut     =  %d\n", es_event->TimeOut);
 | |
| 		break;
 | |
| 	}
 | |
| 	case UPNP_EVENT_AUTORENEWAL_FAILED:
 | |
| 	case UPNP_EVENT_SUBSCRIPTION_EXPIRED: {
 | |
| 		struct Upnp_Event_Subscribe *es_event =
 | |
| 			(struct Upnp_Event_Subscribe *)Event;
 | |
| 
 | |
| 		SampleUtil_Print("SID         =  %s\n", es_event->Sid);
 | |
| 		SampleUtil_Print("ErrCode     =  %s(%d)\n",  
 | |
| 			UpnpGetErrorMessage(es_event->ErrCode), es_event->ErrCode);
 | |
| 		SampleUtil_Print("PublisherURL=  %s\n", es_event->PublisherUrl);
 | |
| 		SampleUtil_Print("TimeOut     =  %d\n", es_event->TimeOut);
 | |
| 		break;
 | |
| 	}
 | |
| 	}
 | |
| 	SampleUtil_Print(
 | |
| 		"----------------------------------------------------------------------\n"
 | |
| 		"======================================================================\n"
 | |
| 		"\n\n\n");
 | |
| 
 | |
| 	ithread_mutex_unlock(&display_mutex);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| int SampleUtil_FindAndParseService(IXML_Document *DescDoc, const char *location,
 | |
| 	char *serviceType, char **serviceId, char **eventURL, char **controlURL)
 | |
| {
 | |
| 	int i;
 | |
| 	int length;
 | |
| 	int found = 0;
 | |
| 	int ret;
 | |
| 	int sindex = 0;
 | |
| 	char *tempServiceType = NULL;
 | |
| 	char *baseURL = NULL;
 | |
| 	const char *base = NULL;
 | |
| 	char *relcontrolURL = NULL;
 | |
| 	char *releventURL = NULL;
 | |
| 	IXML_NodeList *serviceList = NULL;
 | |
| 	IXML_Element *service = NULL;
 | |
| 
 | |
| 	baseURL = SampleUtil_GetFirstDocumentItem(DescDoc, "URLBase");
 | |
| 	if (baseURL) {
 | |
| 		base = baseURL;
 | |
| 	} else {
 | |
| 		base = location;
 | |
| 	}
 | |
| 
 | |
| 	/* Top level */
 | |
| 	for (sindex = 0;
 | |
| 	     (serviceList = SampleUtil_GetNthServiceList(DescDoc , sindex)) != NULL;
 | |
| 	     sindex ++) {
 | |
| 		tempServiceType = NULL;
 | |
| 		relcontrolURL = NULL;
 | |
| 		releventURL = NULL;
 | |
| 		service = NULL;
 | |
| 
 | |
| 		/* serviceList = SampleUtil_GetFirstServiceList( DescDoc ); */
 | |
| 		length = ixmlNodeList_length(serviceList);
 | |
| 		for (i = 0; i < length; i++) {
 | |
| 			service = (IXML_Element *)ixmlNodeList_item(serviceList, i);
 | |
| 			tempServiceType =
 | |
| 				SampleUtil_GetFirstElementItem(
 | |
| 					(IXML_Element *)service, "serviceType");
 | |
| 			if (strcmp(tempServiceType, serviceType) == 0) {
 | |
| 				SampleUtil_Print("Found service: %s\n", serviceType);
 | |
| 				*serviceId =
 | |
| 					SampleUtil_GetFirstElementItem(service, "serviceId");
 | |
| 				SampleUtil_Print("serviceId: %s\n", *serviceId);
 | |
| 				relcontrolURL =
 | |
| 					SampleUtil_GetFirstElementItem(service, "controlURL");
 | |
| 				releventURL =
 | |
| 					SampleUtil_GetFirstElementItem(service, "eventSubURL");
 | |
| 				*controlURL =
 | |
| 					malloc(strlen(base) + strlen(relcontrolURL)+1);
 | |
| 				if (*controlURL) {
 | |
| 					ret = UpnpResolveURL(base, relcontrolURL, *controlURL);
 | |
| 					if (ret != UPNP_E_SUCCESS) {
 | |
| 						SampleUtil_Print("Error generating controlURL from %s + %s\n",
 | |
| 							base, relcontrolURL);
 | |
| 					}
 | |
| 				}
 | |
| 				*eventURL = malloc(strlen(base) + strlen(releventURL)+1);
 | |
| 				if (*eventURL) {
 | |
| 					ret = UpnpResolveURL(base, releventURL, *eventURL);
 | |
| 					if (ret != UPNP_E_SUCCESS) {
 | |
| 						SampleUtil_Print("Error generating eventURL from %s + %s\n",
 | |
| 							base, releventURL);
 | |
| 					}
 | |
| 				}
 | |
| 				free(relcontrolURL);
 | |
| 				free(releventURL);
 | |
| 				relcontrolURL = NULL;
 | |
| 				releventURL = NULL;
 | |
| 				found = 1;
 | |
| 				break;
 | |
| 			}
 | |
| 			free(tempServiceType);
 | |
| 			tempServiceType = NULL;
 | |
| 		}
 | |
| 		free(tempServiceType);
 | |
| 		tempServiceType = NULL;
 | |
| 		if (serviceList) {
 | |
| 			ixmlNodeList_free(serviceList);
 | |
| 		}
 | |
| 		serviceList = NULL;
 | |
| 	}
 | |
| 	free(baseURL);
 | |
| 
 | |
| 	return found;
 | |
| }
 | |
| 
 | |
| int SampleUtil_Print(const char *fmt, ...)
 | |
| {
 | |
| #define MAX_BUF (8 * 1024)
 | |
| 	va_list ap;
 | |
| 	static char buf[MAX_BUF];
 | |
| 	int rc;
 | |
| 
 | |
| 	/* Protect both the display and the static buffer with the mutex */
 | |
| 	ithread_mutex_lock(&display_mutex);
 | |
| 
 | |
| 	va_start(ap, fmt);
 | |
| 	rc = vsnprintf(buf, MAX_BUF, fmt, ap);
 | |
| 	va_end(ap);
 | |
| 
 | |
| 	if (gPrintFun) {
 | |
| 		gPrintFun(buf);
 | |
| 	}
 | |
| 
 | |
| 	ithread_mutex_unlock(&display_mutex);
 | |
| 
 | |
| 	return rc;
 | |
| }
 | |
| 
 | |
| void SampleUtil_StateUpdate(const char *varName, const char *varValue,
 | |
| 	const char *UDN, eventType type)
 | |
| {
 | |
| 	/* TBD: Add mutex here? */
 | |
| 	if (gStateUpdateFun) {
 | |
| 		gStateUpdateFun(varName, varValue, UDN, type);
 | |
| 	}
 | |
| }
 | |
| 
 | 
