libupnp/upnp/sample/tvcombo/upnp_tv_device.c
Marcelo Roberto Jimenez 4b40e94b03 Merge of trunk into branch 1.6.x.
git-svn-id: https://pupnp.svn.sourceforge.net/svnroot/pupnp/branches/branch-1.6.x@339 119443c7-1b9e-41f8-b6fc-b9c35fce742c
2008-03-25 10:21:08 +00:00

2039 lines
63 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 "upnp_tv_device.h"
#define DEFAULT_WEB_DIR "./web"
#define DESC_URL_SIZE 200
/*
Device type for tv device
*/
extern char TvDeviceType[];
/*
Service types for tv services
*/
extern char *TvServiceType[];
/*
Global arrays for storing Tv Control Service
variable names, values, and defaults
*/
char *tvc_varname[] = { "Power", "Channel", "Volume" };
char tvc_varval[TV_CONTROL_VARCOUNT][TV_MAX_VAL_LEN];
char *tvc_varval_def[] = { "1", "1", "5" };
/*
Global arrays for storing Tv Picture Service
variable names, values, and defaults
*/
char *tvp_varname[] = { "Color", "Tint", "Contrast", "Brightness" };
char tvp_varval[TV_PICTURE_VARCOUNT][TV_MAX_VAL_LEN];
char *tvp_varval_def[] = { "5", "5", "5", "5" };
/*
The amount of time (in seconds) before advertisements
will expire
*/
int default_advr_expire = 100;
/*
Global structure for storing the state table for this device
*/
struct TvService tv_service_table[2];
/*
Device handle supplied by UPnP SDK
*/
UpnpDevice_Handle device_handle = -1;
/*
Mutex for protecting the global state table data
in a multi-threaded, asynchronous environment.
All functions should lock this mutex before reading
or writing the state table data.
*/
ithread_mutex_t TVDevMutex;
//Color constants
#define MAX_COLOR 10
#define MIN_COLOR 1
//Brightness constants
#define MAX_BRIGHTNESS 10
#define MIN_BRIGHTNESS 1
//Power constants
#define POWER_ON 1
#define POWER_OFF 0
//Tint constants
#define MAX_TINT 10
#define MIN_TINT 1
//Volume constants
#define MAX_VOLUME 10
#define MIN_VOLUME 1
//Contrast constants
#define MAX_CONTRAST 10
#define MIN_CONTRAST 1
//Channel constants
#define MAX_CHANNEL 100
#define MIN_CHANNEL 1
/******************************************************************************
* SetServiceTable
*
* Description:
* Initializes the service table for the specified service.
* Note that
* knowledge of the service description is
* assumed.
* Parameters:
* int serviceType - one of TV_SERVICE_CONTROL or, TV_SERVICE_PICTURE
* const char * UDN - UDN of device containing service
* const char * serviceId - serviceId of service
* const char * serviceTypeS - service type (as specified in Description
* Document)
* struct TvService *out - service containing table to be set.
*
*****************************************************************************/
int
SetServiceTable( IN int serviceType,
IN const char *UDN,
IN const char *serviceId,
IN const char *serviceTypeS,
INOUT struct TvService *out )
{
unsigned int i = 0;
strcpy( out->UDN, UDN );
strcpy( out->ServiceId, serviceId );
strcpy( out->ServiceType, serviceTypeS );
switch ( serviceType ) {
case TV_SERVICE_CONTROL:
out->VariableCount = TV_CONTROL_VARCOUNT;
for( i = 0;
i < tv_service_table[TV_SERVICE_CONTROL].VariableCount;
i++ ) {
tv_service_table[TV_SERVICE_CONTROL].VariableName[i]
= tvc_varname[i];
tv_service_table[TV_SERVICE_CONTROL].VariableStrVal[i]
= tvc_varval[i];
strcpy( tv_service_table[TV_SERVICE_CONTROL].
VariableStrVal[i], tvc_varval_def[i] );
}
break;
case TV_SERVICE_PICTURE:
out->VariableCount = TV_PICTURE_VARCOUNT;
for( i = 0;
i < tv_service_table[TV_SERVICE_PICTURE].VariableCount;
i++ ) {
tv_service_table[TV_SERVICE_PICTURE].VariableName[i] =
tvp_varname[i];
tv_service_table[TV_SERVICE_PICTURE].VariableStrVal[i] =
tvp_varval[i];
strcpy( tv_service_table[TV_SERVICE_PICTURE].
VariableStrVal[i], tvp_varval_def[i] );
}
break;
default:
assert( 0 );
}
return SetActionTable( serviceType, out );
}
/******************************************************************************
* SetActionTable
*
* Description:
* Initializes the action table for the specified service.
* Note that
* knowledge of the service description is
* assumed. Action names are hardcoded.
* Parameters:
* int serviceType - one of TV_SERVICE_CONTROL or, TV_SERVICE_PICTURE
* struct TvService *out - service containing action table to set.
*
*****************************************************************************/
int
SetActionTable( IN int serviceType,
INOUT struct TvService *out )
{
if( serviceType == TV_SERVICE_CONTROL ) {
out->ActionNames[0] = "PowerOn";
out->actions[0] = TvDevicePowerOn;
out->ActionNames[1] = "PowerOff";
out->actions[1] = TvDevicePowerOff;
out->ActionNames[2] = "SetChannel";
out->actions[2] = TvDeviceSetChannel;
out->ActionNames[3] = "IncreaseChannel";
out->actions[3] = TvDeviceIncreaseChannel;
out->ActionNames[4] = "DecreaseChannel";
out->actions[4] = TvDeviceDecreaseChannel;
out->ActionNames[5] = "SetVolume";
out->actions[5] = TvDeviceSetVolume;
out->ActionNames[6] = "IncreaseVolume";
out->actions[6] = TvDeviceIncreaseVolume;
out->ActionNames[7] = "DecreaseVolume";
out->actions[7] = TvDeviceDecreaseVolume;
out->ActionNames[8] = NULL;
return 1;
} else if( serviceType == TV_SERVICE_PICTURE ) {
out->ActionNames[0] = "SetColor";
out->ActionNames[1] = "IncreaseColor";
out->ActionNames[2] = "DecreaseColor";
out->actions[0] = TvDeviceSetColor;
out->actions[1] = TvDeviceIncreaseColor;
out->actions[2] = TvDeviceDecreaseColor;
out->ActionNames[3] = "SetTint";
out->ActionNames[4] = "IncreaseTint";
out->ActionNames[5] = "DecreaseTint";
out->actions[3] = TvDeviceSetTint;
out->actions[4] = TvDeviceIncreaseTint;
out->actions[5] = TvDeviceDecreaseTint;
out->ActionNames[6] = "SetBrightness";
out->ActionNames[7] = "IncreaseBrightness";
out->ActionNames[8] = "DecreaseBrightness";
out->actions[6] = TvDeviceSetBrightness;
out->actions[7] = TvDeviceIncreaseBrightness;
out->actions[8] = TvDeviceDecreaseBrightness;
out->ActionNames[9] = "SetContrast";
out->ActionNames[10] = "IncreaseContrast";
out->ActionNames[11] = "DecreaseContrast";
out->actions[9] = TvDeviceSetContrast;
out->actions[10] = TvDeviceIncreaseContrast;
out->actions[11] = TvDeviceDecreaseContrast;
return 1;
}
return 0;
}
/******************************************************************************
* TvDeviceStateTableInit
*
* Description:
* Initialize the device state table for
* this TvDevice, pulling identifier info
* from the description Document. Note that
* knowledge of the service description is
* assumed. State table variables and default
* values are currently hardcoded in this file
* rather than being read from service description
* documents.
*
* Parameters:
* DescDocURL -- The description document URL
*
*****************************************************************************/
int
TvDeviceStateTableInit( IN char *DescDocURL )
{
IXML_Document *DescDoc = NULL;
int ret = UPNP_E_SUCCESS;
char *servid_ctrl = NULL,
*evnturl_ctrl = NULL,
*ctrlurl_ctrl = NULL;
char *servid_pict = NULL,
*evnturl_pict = NULL,
*ctrlurl_pict = NULL;
char *udn = NULL;
//Download description document
if( UpnpDownloadXmlDoc( DescDocURL, &DescDoc ) != UPNP_E_SUCCESS ) {
SampleUtil_Print( "TvDeviceStateTableInit -- Error Parsing %s\n",
DescDocURL );
ret = UPNP_E_INVALID_DESC;
goto error_handler;
}
udn = SampleUtil_GetFirstDocumentItem( DescDoc, "UDN" );
/*
Find the Tv Control Service identifiers
*/
if( !SampleUtil_FindAndParseService( DescDoc, DescDocURL,
TvServiceType[TV_SERVICE_CONTROL],
&servid_ctrl, &evnturl_ctrl,
&ctrlurl_ctrl ) ) {
SampleUtil_Print( "TvDeviceStateTableInit -- Error: Could not find"
" Service: %s\n",
TvServiceType[TV_SERVICE_CONTROL] );
ret = UPNP_E_INVALID_DESC;
goto error_handler;
}
//set control service table
SetServiceTable( TV_SERVICE_CONTROL, udn, servid_ctrl,
TvServiceType[TV_SERVICE_CONTROL],
&tv_service_table[TV_SERVICE_CONTROL] );
/*
Find the Tv Picture Service identifiers
*/
if( !SampleUtil_FindAndParseService( DescDoc, DescDocURL,
TvServiceType[TV_SERVICE_PICTURE],
&servid_pict, &evnturl_pict,
&ctrlurl_pict ) ) {
SampleUtil_Print( "TvDeviceStateTableInit -- Error: Could not find"
" Service: %s\n",
TvServiceType[TV_SERVICE_PICTURE] );
ret = UPNP_E_INVALID_DESC;
goto error_handler;
}
//set picture service table
SetServiceTable( TV_SERVICE_PICTURE, udn, servid_pict,
TvServiceType[TV_SERVICE_PICTURE],
&tv_service_table[TV_SERVICE_PICTURE] );
error_handler:
//clean up
if( udn )
free( udn );
if( servid_ctrl )
free( servid_ctrl );
if( evnturl_ctrl )
free( evnturl_ctrl );
if( ctrlurl_ctrl )
free( ctrlurl_ctrl );
if( servid_pict )
free( servid_pict );
if( evnturl_pict )
free( evnturl_pict );
if( ctrlurl_pict )
free( ctrlurl_pict );
if( DescDoc )
ixmlDocument_free( DescDoc );
return ( ret );
}
/******************************************************************************
* TvDeviceHandleSubscriptionRequest
*
* Description:
* Called during a subscription request callback. If the
* subscription request is for this device and either its
* control service or picture service, then accept it.
*
* Parameters:
* sr_event -- The subscription request event structure
*
*****************************************************************************/
int
TvDeviceHandleSubscriptionRequest( IN struct Upnp_Subscription_Request
*sr_event )
{
unsigned int i = 0; //,j=0;
// IXML_Document *PropSet=NULL;
//lock state mutex
ithread_mutex_lock( &TVDevMutex );
for( i = 0; i < TV_SERVICE_SERVCOUNT; i++ ) {
if( ( strcmp( sr_event->UDN, tv_service_table[i].UDN ) == 0 ) &&
( strcmp( sr_event->ServiceId, tv_service_table[i].ServiceId )
== 0 ) ) {
/*
PropSet = NULL;
for (j=0; j< tv_service_table[i].VariableCount; j++)
{
//add each variable to the property set
//for initial state dump
UpnpAddToPropertySet(&PropSet,
tv_service_table[i].VariableName[j],
tv_service_table[i].VariableStrVal[j]);
}
//dump initial state
UpnpAcceptSubscriptionExt(device_handle, sr_event->UDN,
sr_event->ServiceId,
PropSet,sr_event->Sid);
//free document
Document_free(PropSet);
*/
UpnpAcceptSubscription( device_handle,
sr_event->UDN,
sr_event->ServiceId,
( const char ** )tv_service_table[i].
VariableName,
( const char ** )tv_service_table[i].
VariableStrVal,
tv_service_table[i].VariableCount,
sr_event->Sid );
}
}
ithread_mutex_unlock( &TVDevMutex );
return ( 1 );
}
/******************************************************************************
* TvDeviceHandleGetVarRequest
*
* Description:
* Called during a get variable request callback. If the
* request is for this device and either its control service
* or picture service, then respond with the variable value.
*
* Parameters:
* cgv_event -- The control get variable request event structure
*
*****************************************************************************/
int
TvDeviceHandleGetVarRequest( INOUT struct Upnp_State_Var_Request
*cgv_event )
{
unsigned int i = 0,
j = 0;
int getvar_succeeded = 0;
cgv_event->CurrentVal = NULL;
ithread_mutex_lock( &TVDevMutex );
for( i = 0; i < TV_SERVICE_SERVCOUNT; i++ ) {
//check udn and service id
if( ( strcmp( cgv_event->DevUDN, tv_service_table[i].UDN ) == 0 )
&&
( strcmp( cgv_event->ServiceID, tv_service_table[i].ServiceId )
== 0 ) ) {
//check variable name
for( j = 0; j < tv_service_table[i].VariableCount; j++ ) {
if( strcmp( cgv_event->StateVarName,
tv_service_table[i].VariableName[j] ) == 0 ) {
getvar_succeeded = 1;
cgv_event->CurrentVal =
ixmlCloneDOMString( tv_service_table[i].
VariableStrVal[j] );
break;
}
}
}
}
if( getvar_succeeded ) {
cgv_event->ErrCode = UPNP_E_SUCCESS;
} else {
SampleUtil_Print
( "Error in UPNP_CONTROL_GET_VAR_REQUEST callback:\n" );
SampleUtil_Print( " Unknown variable name = %s\n",
cgv_event->StateVarName );
cgv_event->ErrCode = 404;
strcpy( cgv_event->ErrStr, "Invalid Variable" );
}
ithread_mutex_unlock( &TVDevMutex );
return ( cgv_event->ErrCode == UPNP_E_SUCCESS );
}
/******************************************************************************
* TvDeviceHandleActionRequest
*
* Description:
* Called during an action request callback. If the
* request is for this device and either its control service
* or picture service, then perform the action and respond.
*
* Parameters:
* ca_event -- The control action request event structure
*
*****************************************************************************/
int
TvDeviceHandleActionRequest( INOUT struct Upnp_Action_Request *ca_event )
{
/*
Defaults if action not found
*/
int action_found = 0;
int i = 0;
int service = -1;
int retCode = 0;
char *errorString = NULL;
ca_event->ErrCode = 0;
ca_event->ActionResult = NULL;
if( ( strcmp( ca_event->DevUDN,
tv_service_table[TV_SERVICE_CONTROL].UDN ) == 0 ) &&
( strcmp
( ca_event->ServiceID,
tv_service_table[TV_SERVICE_CONTROL].ServiceId ) == 0 ) ) {
/*
Request for action in the TvDevice Control Service
*/
service = TV_SERVICE_CONTROL;
} else if( ( strcmp( ca_event->DevUDN,
tv_service_table[TV_SERVICE_PICTURE].UDN ) == 0 )
&&
( strcmp
( ca_event->ServiceID,
tv_service_table[TV_SERVICE_PICTURE].ServiceId ) ==
0 ) ) {
/*
Request for action in the TvDevice Picture Service
*/
service = TV_SERVICE_PICTURE;
}
//Find and call appropriate procedure based on action name
//Each action name has an associated procedure stored in the
//service table. These are set at initialization.
for( i = 0; ( ( i < TV_MAXACTIONS ) &&
( tv_service_table[service].ActionNames[i] != NULL ) );
i++ ) {
if( !strcmp( ca_event->ActionName,
tv_service_table[service].ActionNames[i] ) ) {
if( ( !strcmp( tv_service_table[TV_SERVICE_CONTROL].
VariableStrVal[TV_CONTROL_POWER], "1" ) )
|| ( !strcmp( ca_event->ActionName, "PowerOn" ) ) ) {
retCode =
tv_service_table[service].actions[i] ( ca_event->
ActionRequest,
&ca_event->
ActionResult,
&errorString );
} else {
errorString = "Power is Off";
retCode = UPNP_E_INTERNAL_ERROR;
}
action_found = 1;
break;
}
}
if( !action_found ) {
ca_event->ActionResult = NULL;
strcpy( ca_event->ErrStr, "Invalid Action" );
ca_event->ErrCode = 401;
} else {
if( retCode == UPNP_E_SUCCESS ) {
ca_event->ErrCode = UPNP_E_SUCCESS;
} else {
//copy the error string
strcpy( ca_event->ErrStr, errorString );
switch ( retCode ) {
case UPNP_E_INVALID_PARAM:
{
ca_event->ErrCode = 402;
break;
}
case UPNP_E_INTERNAL_ERROR:
default:
{
ca_event->ErrCode = 501;
break;
}
}
}
}
return ( ca_event->ErrCode );
}
/******************************************************************************
* TvDeviceSetServiceTableVar
*
* Description:
* Update the TvDevice service state table, and notify all subscribed
* control points of the updated state. Note that since this function
* blocks on the mutex TVDevMutex, to avoid a hang this function should
* not be called within any other function that currently has this mutex
* locked.
*
* Parameters:
* service -- The service number (TV_SERVICE_CONTROL or TV_SERVICE_PICTURE)
* variable -- The variable number (TV_CONTROL_POWER, TV_CONTROL_CHANNEL,
* TV_CONTROL_VOLUME, TV_PICTURE_COLOR, TV_PICTURE_TINT,
* TV_PICTURE_CONTRAST, or TV_PICTURE_BRIGHTNESS)
* value -- The string representation of the new value
*
*****************************************************************************/
int
TvDeviceSetServiceTableVar( IN unsigned int service,
IN unsigned int variable,
IN char *value )
{
//IXML_Document *PropSet= NULL;
if( ( service >= TV_SERVICE_SERVCOUNT )
|| ( variable >= tv_service_table[service].VariableCount )
|| ( strlen( value ) >= TV_MAX_VAL_LEN ) ) {
return ( 0 );
}
ithread_mutex_lock( &TVDevMutex );
strcpy( tv_service_table[service].VariableStrVal[variable], value );
/*
//Using utility api
PropSet= UpnpCreatePropertySet(1,tv_service_table[service].
VariableName[variable],
tv_service_table[service].
VariableStrVal[variable]);
UpnpNotifyExt(device_handle, tv_service_table[service].UDN,
tv_service_table[service].ServiceId,PropSet);
//Free created property set
Document_free(PropSet);
*/
UpnpNotify( device_handle,
tv_service_table[service].UDN,
tv_service_table[service].ServiceId,
( const char ** )&tv_service_table[service].
VariableName[variable],
( const char ** )&tv_service_table[service].
VariableStrVal[variable], 1 );
ithread_mutex_unlock( &TVDevMutex );
return ( 1 );
}
/******************************************************************************
* TvDeviceSetPower
*
* Description:
* Turn the power on/off, update the TvDevice control service
* state table, and notify all subscribed control points of the
* updated state.
*
* Parameters:
* on -- If 1, turn power on. If 0, turn power off.
*
*****************************************************************************/
int
TvDeviceSetPower( IN int on )
{
char value[TV_MAX_VAL_LEN];
int ret = 0;
if( on != POWER_ON && on != POWER_OFF ) {
SampleUtil_Print( "error: can't set power to value %d\n", on );
return ( 0 );
}
/*
Vendor-specific code to turn the power on/off goes here
*/
sprintf( value, "%d", on );
ret = TvDeviceSetServiceTableVar( TV_SERVICE_CONTROL, TV_CONTROL_POWER,
value );
return ( ret );
}
/******************************************************************************
* TvDevicePowerOn
*
* Description:
* Turn the power on.
*
* Parameters:
*
* IXML_Document * in - document of action request
* IXML_Document **out - action result
* char **errorString - errorString (in case action was unsuccessful)
*
*****************************************************************************/
int
TvDevicePowerOn( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
( *out ) = NULL;
( *errorString ) = NULL;
if( TvDeviceSetPower( POWER_ON ) ) {
//create a response
if( UpnpAddToActionResponse( out, "PowerOn",
TvServiceType[TV_SERVICE_CONTROL],
"Power", "1" ) != UPNP_E_SUCCESS ) {
( *out ) = NULL;
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
return UPNP_E_SUCCESS;
} else {
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
}
/******************************************************************************
* TvDevicePowerOff
*
* Description:
* Turn the power off.
*
* Parameters:
*
* IXML_Document * in - document of action request
* IXML_Document **out - action result
* char **errorString - errorString (in case action was unsuccessful)
*
*****************************************************************************/
int
TvDevicePowerOff( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
( *out ) = NULL;
( *errorString ) = NULL;
if( TvDeviceSetPower( POWER_OFF ) ) {
//create a response
if( UpnpAddToActionResponse( out, "PowerOff",
TvServiceType[TV_SERVICE_CONTROL],
"Power", "0" ) != UPNP_E_SUCCESS ) {
( *out ) = NULL;
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
return UPNP_E_SUCCESS;
}
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
/******************************************************************************
* TvDeviceSetChannel
*
* Description:
* Change the channel, update the TvDevice control service
* state table, and notify all subscribed control points of the
* updated state.
*
* Parameters:
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*
*****************************************************************************/
int
TvDeviceSetChannel( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
char *value = NULL;
int channel = 0;
( *out ) = NULL;
( *errorString ) = NULL;
if( !( value = SampleUtil_GetFirstDocumentItem( in, "Channel" ) ) ) {
( *errorString ) = "Invalid Channel";
return UPNP_E_INVALID_PARAM;
}
channel = atoi( value );
if( channel < MIN_CHANNEL || channel > MAX_CHANNEL ) {
free( value );
SampleUtil_Print( "error: can't change to channel %d\n", channel );
( *errorString ) = "Invalid Channel";
return UPNP_E_INVALID_PARAM;
}
/*
Vendor-specific code to set the channel goes here
*/
if( TvDeviceSetServiceTableVar( TV_SERVICE_CONTROL,
TV_CONTROL_CHANNEL, value ) ) {
if( UpnpAddToActionResponse( out, "SetChannel",
TvServiceType[TV_SERVICE_CONTROL],
"NewChannel",
value ) != UPNP_E_SUCCESS ) {
( *out ) = NULL;
( *errorString ) = "Internal Error";
free( value );
return UPNP_E_INTERNAL_ERROR;
}
free( value );
return UPNP_E_SUCCESS;
} else {
free( value );
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
}
/******************************************************************************
* IncrementChannel
*
* Description:
* Increment the channel. Read the current channel from the state
* table, add the increment, and then change the channel.
*
* Parameters:
* incr -- The increment by which to change the channel.
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*****************************************************************************/
int
IncrementChannel( IN int incr,
IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
int curchannel,
newchannel;
char *actionName = NULL;
char value[TV_MAX_VAL_LEN];
if( incr > 0 ) {
actionName = "IncreaseChannel";
} else {
actionName = "DecreaseChannel";
}
ithread_mutex_lock( &TVDevMutex );
curchannel = atoi( tv_service_table[TV_SERVICE_CONTROL].
VariableStrVal[TV_CONTROL_CHANNEL] );
ithread_mutex_unlock( &TVDevMutex );
newchannel = curchannel + incr;
if( newchannel < MIN_CHANNEL || newchannel > MAX_CHANNEL ) {
SampleUtil_Print( "error: can't change to channel %d\n",
newchannel );
( *errorString ) = "Invalid Channel";
return UPNP_E_INVALID_PARAM;
}
/*
Vendor-specific code to set the channel goes here
*/
sprintf( value, "%d", newchannel );
if( TvDeviceSetServiceTableVar( TV_SERVICE_CONTROL,
TV_CONTROL_CHANNEL, value ) ) {
if( UpnpAddToActionResponse( out, actionName,
TvServiceType[TV_SERVICE_CONTROL],
"Channel", value ) != UPNP_E_SUCCESS )
{
( *out ) = NULL;
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
return UPNP_E_SUCCESS;
} else {
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
}
/******************************************************************************
* TvDeviceDecreaseChannel
*
* Description:
* Decrease the channel.
*
* Parameters:
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*
*****************************************************************************/
int
TvDeviceDecreaseChannel( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
return IncrementChannel( -1, in, out, errorString );
}
/******************************************************************************
* TvDeviceIncreaseChannel
*
* Description:
* Increase the channel.
*
* Parameters:
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*
*****************************************************************************/
int
TvDeviceIncreaseChannel( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
return IncrementChannel( 1, in, out, errorString );
}
/******************************************************************************
* TvDeviceSetVolume
*
* Description:
* Change the volume, update the TvDevice control service
* state table, and notify all subscribed control points of the
* updated state.
*
* Parameters:
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*
*****************************************************************************/
int
TvDeviceSetVolume( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
char *value = NULL;
int volume = 0;
( *out ) = NULL;
( *errorString ) = NULL;
if( !( value = SampleUtil_GetFirstDocumentItem( in, "Volume" ) ) ) {
( *errorString ) = "Invalid Volume";
return UPNP_E_INVALID_PARAM;
}
volume = atoi( value );
if( volume < MIN_VOLUME || volume > MAX_VOLUME ) {
SampleUtil_Print( "error: can't change to volume %d\n", volume );
( *errorString ) = "Invalid Volume";
return UPNP_E_INVALID_PARAM;
}
/*
Vendor-specific code to set the volume goes here
*/
if( TvDeviceSetServiceTableVar( TV_SERVICE_CONTROL,
TV_CONTROL_VOLUME, value ) ) {
if( UpnpAddToActionResponse( out, "SetVolume",
TvServiceType[TV_SERVICE_CONTROL],
"NewVolume",
value ) != UPNP_E_SUCCESS ) {
( *out ) = NULL;
( *errorString ) = "Internal Error";
free( value );
return UPNP_E_INTERNAL_ERROR;
}
free( value );
return UPNP_E_SUCCESS;
} else {
free( value );
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
}
/******************************************************************************
* IncrementVolume
*
* Description:
* Increment the volume. Read the current volume from the state
* table, add the increment, and then change the volume.
*
* Parameters:
* incr -- The increment by which to change the volume.
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*
*****************************************************************************/
int
IncrementVolume( IN int incr,
IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
int curvolume,
newvolume;
char *actionName = NULL;
char value[TV_MAX_VAL_LEN];
if( incr > 0 ) {
actionName = "IncreaseVolume";
} else {
actionName = "DecreaseVolume";
}
ithread_mutex_lock( &TVDevMutex );
curvolume = atoi( tv_service_table[TV_SERVICE_CONTROL].
VariableStrVal[TV_CONTROL_VOLUME] );
ithread_mutex_unlock( &TVDevMutex );
newvolume = curvolume + incr;
if( newvolume < MIN_VOLUME || newvolume > MAX_VOLUME ) {
SampleUtil_Print( "error: can't change to volume %d\n",
newvolume );
( *errorString ) = "Invalid Volume";
return UPNP_E_INVALID_PARAM;
}
/*
Vendor-specific code to set the channel goes here
*/
sprintf( value, "%d", newvolume );
if( TvDeviceSetServiceTableVar( TV_SERVICE_CONTROL,
TV_CONTROL_VOLUME, value ) ) {
if( UpnpAddToActionResponse( out, actionName,
TvServiceType[TV_SERVICE_CONTROL],
"Volume", value ) != UPNP_E_SUCCESS )
{
( *out ) = NULL;
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
return UPNP_E_SUCCESS;
} else {
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
}
/******************************************************************************
* TvDeviceIncrVolume
*
* Description:
* Increase the volume.
*
* Parameters:
*
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*****************************************************************************/
int
TvDeviceIncreaseVolume( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
return IncrementVolume( 1, in, out, errorString );
}
/******************************************************************************
* TvDeviceDecreaseVolume
*
* Description:
* Decrease the volume.
*
* Parameters:
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*
*****************************************************************************/
int
TvDeviceDecreaseVolume( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
return IncrementVolume( -1, in, out, errorString );
}
/******************************************************************************
* TvDeviceSetColor
*
* Description:
* Change the color, update the TvDevice picture service
* state table, and notify all subscribed control points of the
* updated state.
*
* Parameters:
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*
*****************************************************************************/
int
TvDeviceSetColor( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
char *value = NULL;
int color = 0;
( *out ) = NULL;
( *errorString ) = NULL;
if( !( value = SampleUtil_GetFirstDocumentItem( in, "Color" ) ) ) {
( *errorString ) = "Invalid Color";
return UPNP_E_INVALID_PARAM;
}
color = atoi( value );
if( color < MIN_COLOR || color > MAX_COLOR ) {
SampleUtil_Print( "error: can't change to color %d\n", color );
( *errorString ) = "Invalid Color";
return UPNP_E_INVALID_PARAM;
}
/*
Vendor-specific code to set the volume goes here
*/
if( TvDeviceSetServiceTableVar( TV_SERVICE_PICTURE,
TV_PICTURE_COLOR, value ) ) {
if( UpnpAddToActionResponse( out, "SetColor",
TvServiceType[TV_SERVICE_PICTURE],
"NewColor",
value ) != UPNP_E_SUCCESS ) {
( *out ) = NULL;
( *errorString ) = "Internal Error";
free( value );
return UPNP_E_INTERNAL_ERROR;
}
free( value );
return UPNP_E_SUCCESS;
} else {
free( value );
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
}
/******************************************************************************
* IncrementColor
*
* Description:
* Increment the color. Read the current color from the state
* table, add the increment, and then change the color.
*
* Parameters:
* incr -- The increment by which to change the color.
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*****************************************************************************/
int
IncrementColor( IN int incr,
IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
int curcolor,
newcolor;
char *actionName;
char value[TV_MAX_VAL_LEN];
if( incr > 0 ) {
actionName = "IncreaseColor";
} else {
actionName = "DecreaseColor";
}
ithread_mutex_lock( &TVDevMutex );
curcolor = atoi( tv_service_table[TV_SERVICE_PICTURE].
VariableStrVal[TV_PICTURE_COLOR] );
ithread_mutex_unlock( &TVDevMutex );
newcolor = curcolor + incr;
if( newcolor < MIN_COLOR || newcolor > MAX_COLOR ) {
SampleUtil_Print( "error: can't change to color %d\n", newcolor );
( *errorString ) = "Invalid Color";
return UPNP_E_INVALID_PARAM;
}
/*
Vendor-specific code to set the channel goes here
*/
sprintf( value, "%d", newcolor );
if( TvDeviceSetServiceTableVar( TV_SERVICE_PICTURE,
TV_PICTURE_COLOR, value ) ) {
if( UpnpAddToActionResponse( out, actionName,
TvServiceType[TV_SERVICE_PICTURE],
"Color", value ) != UPNP_E_SUCCESS ) {
( *out ) = NULL;
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
return UPNP_E_SUCCESS;
} else {
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
}
/******************************************************************************
* TvDeviceDecreaseColor
*
* Description:
* Decrease the color.
*
* Parameters:
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*****************************************************************************/
int
TvDeviceDecreaseColor( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
return IncrementColor( -1, in, out, errorString );
}
/******************************************************************************
* TvDeviceIncreaseColor
*
* Description:
* Increase the color.
*
* Parameters:
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*****************************************************************************/
int
TvDeviceIncreaseColor( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
return IncrementColor( 1, in, out, errorString );
}
/******************************************************************************
* TvDeviceSetTint
*
* Description:
* Change the tint, update the TvDevice picture service
* state table, and notify all subscribed control points of the
* updated state.
*
* Parameters:
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*
*****************************************************************************/
int
TvDeviceSetTint( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
char *value = NULL;
int tint = -1;
( *out ) = NULL;
( *errorString ) = NULL;
if( !( value = SampleUtil_GetFirstDocumentItem( in, "Tint" ) ) ) {
( *errorString ) = "Invalid Tint";
return UPNP_E_INVALID_PARAM;
}
tint = atoi( value );
if( tint < MIN_TINT || tint > MAX_TINT ) {
SampleUtil_Print( "error: can't change to tint %d\n", tint );
( *errorString ) = "Invalid Tint";
return UPNP_E_INVALID_PARAM;
}
/*
Vendor-specific code to set the volume goes here
*/
if( TvDeviceSetServiceTableVar( TV_SERVICE_PICTURE,
TV_PICTURE_TINT, value ) ) {
if( UpnpAddToActionResponse( out, "SetTint",
TvServiceType[TV_SERVICE_PICTURE],
"NewTint", value ) != UPNP_E_SUCCESS )
{
( *out ) = NULL;
( *errorString ) = "Internal Error";
free( value );
return UPNP_E_INTERNAL_ERROR;
}
free( value );
return UPNP_E_SUCCESS;
} else {
free( value );
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
}
/******************************************************************************
* IncrementTint
*
* Description:
* Increment the tint. Read the current tint from the state
* table, add the increment, and then change the tint.
*
* Parameters:
* incr -- The increment by which to change the tint.
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*****************************************************************************/
int
IncrementTint( IN int incr,
IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
int curtint,
newtint;
char *actionName = NULL;
char value[TV_MAX_VAL_LEN];
if( incr > 0 ) {
actionName = "IncreaseTint";
} else {
actionName = "DecreaseTint";
}
ithread_mutex_lock( &TVDevMutex );
curtint = atoi( tv_service_table[TV_SERVICE_PICTURE].
VariableStrVal[TV_PICTURE_TINT] );
ithread_mutex_unlock( &TVDevMutex );
newtint = curtint + incr;
if( newtint < MIN_TINT || newtint > MAX_TINT ) {
SampleUtil_Print( "error: can't change to tint %d\n", newtint );
( *errorString ) = "Invalid Tint";
return UPNP_E_INVALID_PARAM;
}
/*
Vendor-specific code to set the channel goes here
*/
sprintf( value, "%d", newtint );
if( TvDeviceSetServiceTableVar( TV_SERVICE_PICTURE,
TV_PICTURE_TINT, value ) ) {
if( UpnpAddToActionResponse( out, actionName,
TvServiceType[TV_SERVICE_PICTURE],
"Tint", value ) != UPNP_E_SUCCESS ) {
( *out ) = NULL;
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
return UPNP_E_SUCCESS;
} else {
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
}
/******************************************************************************
* TvDeviceIncreaseTint
*
* Description:
* Increase tint.
*
* Parameters:
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*
*****************************************************************************/
int
TvDeviceIncreaseTint( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
return IncrementTint( 1, in, out, errorString );
}
/******************************************************************************
* TvDeviceDecreaseTint
*
* Description:
* Decrease tint.
*
* Parameters:
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*
*****************************************************************************/
int
TvDeviceDecreaseTint( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
return IncrementTint( -1, in, out, errorString );
}
/*****************************************************************************
* TvDeviceSetContrast
*
* Description:
* Change the contrast, update the TvDevice picture service
* state table, and notify all subscribed control points of the
* updated state.
*
* Parameters:
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*
****************************************************************************/
int
TvDeviceSetContrast( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
char *value = NULL;
int contrast = -1;
( *out ) = NULL;
( *errorString ) = NULL;
if( !( value = SampleUtil_GetFirstDocumentItem( in, "Contrast" ) ) ) {
( *errorString ) = "Invalid Contrast";
return UPNP_E_INVALID_PARAM;
}
contrast = atoi( value );
if( contrast < MIN_CONTRAST || contrast > MAX_CONTRAST ) {
SampleUtil_Print( "error: can't change to contrast %d\n",
contrast );
( *errorString ) = "Invalid Contrast";
return UPNP_E_INVALID_PARAM;
}
/*
Vendor-specific code to set the volume goes here
*/
if( TvDeviceSetServiceTableVar( TV_SERVICE_PICTURE,
TV_PICTURE_CONTRAST, value ) ) {
if( UpnpAddToActionResponse( out, "SetContrast",
TvServiceType[TV_SERVICE_PICTURE],
"NewContrast",
value ) != UPNP_E_SUCCESS ) {
( *out ) = NULL;
( *errorString ) = "Internal Error";
free( value );
return UPNP_E_INTERNAL_ERROR;
}
free( value );
return UPNP_E_SUCCESS;
} else {
free( value );
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
}
/******************************************************************************
* IncrementContrast
*
* Description:
* Increment the contrast. Read the current contrast from the state
* table, add the increment, and then change the contrast.
*
* Parameters:
* incr -- The increment by which to change the contrast.
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*****************************************************************************/
int
IncrementContrast( IN int incr,
IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
int curcontrast,
newcontrast;
char *actionName = NULL;
char value[TV_MAX_VAL_LEN];
if( incr > 0 ) {
actionName = "IncreaseContrast";
} else {
actionName = "DecreaseContrast";
}
ithread_mutex_lock( &TVDevMutex );
curcontrast = atoi( tv_service_table[TV_SERVICE_PICTURE].
VariableStrVal[TV_PICTURE_CONTRAST] );
ithread_mutex_unlock( &TVDevMutex );
newcontrast = curcontrast + incr;
if( newcontrast < MIN_CONTRAST || newcontrast > MAX_CONTRAST ) {
SampleUtil_Print( "error: can't change to contrast %d\n",
newcontrast );
( *errorString ) = "Invalid Contrast";
return UPNP_E_INVALID_PARAM;
}
/*
Vendor-specific code to set the channel goes here
*/
sprintf( value, "%d", newcontrast );
if( TvDeviceSetServiceTableVar( TV_SERVICE_PICTURE,
TV_PICTURE_CONTRAST, value ) ) {
if( UpnpAddToActionResponse( out, actionName,
TvServiceType[TV_SERVICE_PICTURE],
"Contrast",
value ) != UPNP_E_SUCCESS ) {
( *out ) = NULL;
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
return UPNP_E_SUCCESS;
} else {
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
}
/******************************************************************************
* TvDeviceIncreaseContrast
*
* Description:
*
* Increase the contrast.
*
* Parameters:
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*
*****************************************************************************/
int
TvDeviceIncreaseContrast( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
return IncrementContrast( 1, in, out, errorString );
}
/******************************************************************************
* TvDeviceDecreaseContrast
*
* Description:
* Decrease the contrast.
*
* Parameters:
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*
*****************************************************************************/
int
TvDeviceDecreaseContrast( IXML_Document * in,
IXML_Document ** out,
char **errorString )
{
return IncrementContrast( -1, in, out, errorString );
}
/******************************************************************************
* TvDeviceSetBrightness
*
* Description:
* Change the brightness, update the TvDevice picture service
* state table, and notify all subscribed control points of the
* updated state.
*
* Parameters:
* brightness -- The brightness value to change to.
*
*****************************************************************************/
int
TvDeviceSetBrightness( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
char *value = NULL;
int brightness = -1;
( *out ) = NULL;
( *errorString ) = NULL;
if( !( value = SampleUtil_GetFirstDocumentItem( in, "Brightness" ) ) ) {
( *errorString ) = "Invalid Brightness";
return UPNP_E_INVALID_PARAM;
}
brightness = atoi( value );
if( brightness < MIN_BRIGHTNESS || brightness > MAX_BRIGHTNESS ) {
SampleUtil_Print( "error: can't change to brightness %d\n",
brightness );
( *errorString ) = "Invalid Brightness";
return UPNP_E_INVALID_PARAM;
}
/*
Vendor-specific code to set the volume goes here
*/
if( TvDeviceSetServiceTableVar( TV_SERVICE_PICTURE,
TV_PICTURE_BRIGHTNESS, value ) ) {
if( UpnpAddToActionResponse( out, "SetBrightness",
TvServiceType[TV_SERVICE_PICTURE],
"NewBrightness",
value ) != UPNP_E_SUCCESS ) {
( *out ) = NULL;
( *errorString ) = "Internal Error";
free( value );
return UPNP_E_INTERNAL_ERROR;
}
free( value );
return UPNP_E_SUCCESS;
} else {
free( value );
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
}
/******************************************************************************
* IncrementBrightness
*
* Description:
* Increment the brightness. Read the current brightness from the state
* table, add the increment, and then change the brightness.
*
* Parameters:
* incr -- The increment by which to change the brightness.
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*****************************************************************************/
int
IncrementBrightness( IN int incr,
IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
int curbrightness,
newbrightness;
char *actionName = NULL;
char value[TV_MAX_VAL_LEN];
if( incr > 0 ) {
actionName = "IncreaseBrightness";
} else {
actionName = "DecreaseBrightness";
}
ithread_mutex_lock( &TVDevMutex );
curbrightness = atoi( tv_service_table[TV_SERVICE_PICTURE].
VariableStrVal[TV_PICTURE_BRIGHTNESS] );
ithread_mutex_unlock( &TVDevMutex );
newbrightness = curbrightness + incr;
if( newbrightness < MIN_BRIGHTNESS || newbrightness > MAX_BRIGHTNESS ) {
SampleUtil_Print( "error: can't change to brightness %d\n",
newbrightness );
( *errorString ) = "Invalid Brightness";
return UPNP_E_INVALID_PARAM;
}
/*
Vendor-specific code to set the channel goes here
*/
sprintf( value, "%d", newbrightness );
if( TvDeviceSetServiceTableVar( TV_SERVICE_PICTURE,
TV_PICTURE_BRIGHTNESS, value ) ) {
if( UpnpAddToActionResponse( out, actionName,
TvServiceType[TV_SERVICE_PICTURE],
"Brightness",
value ) != UPNP_E_SUCCESS ) {
( *out ) = NULL;
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
return UPNP_E_SUCCESS;
} else {
( *errorString ) = "Internal Error";
return UPNP_E_INTERNAL_ERROR;
}
}
/******************************************************************************
* TvDeviceIncreaseBrightness
*
* Description:
* Increase brightness.
*
* Parameters:
*
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*
*****************************************************************************/
int
TvDeviceIncreaseBrightness( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
return IncrementBrightness( 1, in, out, errorString );
}
/******************************************************************************
* TvDeviceDecreaseBrightness
*
* Description:
* Decrease brightnesss.
*
* Parameters:
* IXML_Document * in - action request document
* IXML_Document **out - action result document
* char **errorString - errorString (in case action was unsuccessful)
*
*****************************************************************************/
int
TvDeviceDecreaseBrightness( IN IXML_Document * in,
OUT IXML_Document ** out,
OUT char **errorString )
{
return IncrementBrightness( -1, in, out, errorString );
}
/******************************************************************************
* TvDeviceCallbackEventHandler
*
* Description:
* The callback handler registered with the SDK while registering
* root device. Dispatches the request to the appropriate procedure
* based on the value of EventType. The four requests handled by the
* device are:
* 1) Event Subscription requests.
* 2) Get Variable requests.
* 3) Action requests.
*
* Parameters:
*
* EventType -- The type of callback event
* Event -- Data structure containing event data
* Cookie -- Optional data specified during callback registration
*
*****************************************************************************/
int
TvDeviceCallbackEventHandler( Upnp_EventType EventType,
void *Event,
void *Cookie )
{
switch ( EventType ) {
case UPNP_EVENT_SUBSCRIPTION_REQUEST:
TvDeviceHandleSubscriptionRequest( ( struct
Upnp_Subscription_Request
* )Event );
break;
case UPNP_CONTROL_GET_VAR_REQUEST:
TvDeviceHandleGetVarRequest( ( struct Upnp_State_Var_Request
* )Event );
break;
case UPNP_CONTROL_ACTION_REQUEST:
TvDeviceHandleActionRequest( ( struct Upnp_Action_Request * )
Event );
break;
/*
ignore these cases, since this is not a control point
*/
case UPNP_DISCOVERY_ADVERTISEMENT_ALIVE:
case UPNP_DISCOVERY_SEARCH_RESULT:
case UPNP_DISCOVERY_SEARCH_TIMEOUT:
case UPNP_DISCOVERY_ADVERTISEMENT_BYEBYE:
case UPNP_CONTROL_ACTION_COMPLETE:
case UPNP_CONTROL_GET_VAR_COMPLETE:
case UPNP_EVENT_RECEIVED:
case UPNP_EVENT_RENEWAL_COMPLETE:
case UPNP_EVENT_SUBSCRIBE_COMPLETE:
case UPNP_EVENT_UNSUBSCRIBE_COMPLETE:
break;
default:
SampleUtil_Print
( "Error in TvDeviceCallbackEventHandler: unknown event type %d\n",
EventType );
}
/*
Print a summary of the event received
*/
SampleUtil_PrintEvent( EventType, Event );
return ( 0 );
}
/******************************************************************************
* TvDeviceStop
*
* Description:
* Stops the device. Uninitializes the sdk.
*
* Parameters:
*
*****************************************************************************/
int
TvDeviceStop()
{
UpnpUnRegisterRootDevice( device_handle );
UpnpFinish();
SampleUtil_Finish();
ithread_mutex_destroy( &TVDevMutex );
return UPNP_E_SUCCESS;
}
/******************************************************************************
* TvDeviceStart
*
* Description:
* Initializes the UPnP Sdk, registers the device, and sends out
* advertisements.
*
* Parameters:
*
* ip_address - ip address to initialize the sdk (may be NULL)
* if null, then the first non null loopback address is used.
* port - port number to initialize the sdk (may be 0)
* if zero, then a random number is used.
* desc_doc_name - name of description document.
* may be NULL. Default is tvcombodesc.xml
* web_dir_path - path of web directory.
* may be NULL. Default is ./web (for Linux) or ../tvdevice/web
* for windows.
* pfun - print function to use.
*
*****************************************************************************/
int
TvDeviceStart( char *ip_address,
unsigned short port,
char *desc_doc_name,
char *web_dir_path,
print_string pfun )
{
int ret = UPNP_E_SUCCESS;
char desc_doc_url[DESC_URL_SIZE];
ithread_mutex_init( &TVDevMutex, NULL );
SampleUtil_Initialize( pfun );
SampleUtil_Print(
"Initializing UPnP Sdk with\n"
"\tipaddress = %s port = %u\n",
ip_address, port );
if( ( ret = UpnpInit( ip_address, port ) ) != UPNP_E_SUCCESS ) {
SampleUtil_Print( "Error with UpnpInit -- %d\n", ret );
UpnpFinish();
return ret;
}
if( ip_address == NULL ) {
ip_address = UpnpGetServerIpAddress();
}
if( port == 0 ) {
port = UpnpGetServerPort();
}
SampleUtil_Print(
"UPnP Initialized\n"
"\tipaddress= %s port = %u\n",
ip_address, port );
if( desc_doc_name == NULL ) {
desc_doc_name = "tvcombodesc.xml";
}
if( web_dir_path == NULL ) {
web_dir_path = DEFAULT_WEB_DIR;
}
snprintf( desc_doc_url, DESC_URL_SIZE, "http://%s:%d/%s", ip_address,
port, desc_doc_name );
SampleUtil_Print( "Specifying the webserver root directory -- %s\n",
web_dir_path );
if( ( ret =
UpnpSetWebServerRootDir( web_dir_path ) ) != UPNP_E_SUCCESS ) {
SampleUtil_Print
( "Error specifying webserver root directory -- %s: %d\n",
web_dir_path, ret );
UpnpFinish();
return ret;
}
SampleUtil_Print(
"Registering the RootDevice\n"
"\t with desc_doc_url: %s\n",
desc_doc_url );
if( ( ret = UpnpRegisterRootDevice( desc_doc_url,
TvDeviceCallbackEventHandler,
&device_handle, &device_handle ) )
!= UPNP_E_SUCCESS ) {
SampleUtil_Print( "Error registering the rootdevice : %d\n", ret );
UpnpFinish();
return ret;
} else {
SampleUtil_Print(
"RootDevice Registered\n"
"Initializing State Table\n");
TvDeviceStateTableInit( desc_doc_url );
SampleUtil_Print("State Table Initialized\n");
if( ( ret =
UpnpSendAdvertisement( device_handle, default_advr_expire ) )
!= UPNP_E_SUCCESS ) {
SampleUtil_Print( "Error sending advertisements : %d\n", ret );
UpnpFinish();
return ret;
}
SampleUtil_Print("Advertisements Sent\n");
}
return UPNP_E_SUCCESS;
}