mirror of
				https://github.com/fairwaves/openbts-2.8.git
				synced 2025-11-04 05:53:17 +00:00 
			
		
		
		
	Changes in SIP logging. * Change "cannot parse" message to "out of place" and downgrade it. * When we get a re-INVITE, log at the CRIT level. git-svn-id: http://wush.net/svn/range/software/public/openbts/trunk@4346 19bc5d8c-e614-43d4-8b26-e1612bc8e597
		
			
				
	
	
		
			502 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			502 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
* Copyright 2008, 2009, 2010m 2011 Free Software Foundation, Inc.
 | 
						|
* Copyright 2011 Range Networks, Inc.
 | 
						|
*
 | 
						|
* This software is distributed under the terms of the GNU Affero Public License.
 | 
						|
* See the COPYING file in the main directory for details.
 | 
						|
*
 | 
						|
* This use of this software may be subject to additional restrictions.
 | 
						|
* See the LEGAL file in the main directory for details.
 | 
						|
 | 
						|
	This program is free software: you can redistribute it and/or modify
 | 
						|
	it under the terms of the GNU Affero General Public License as published by
 | 
						|
	the Free Software Foundation, either version 3 of the License, or
 | 
						|
	(at your option) any later version.
 | 
						|
 | 
						|
	This program is distributed in the hope that it will be useful,
 | 
						|
	but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
	MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
						|
	GNU Affero General Public License for more details.
 | 
						|
 | 
						|
	You should have received a copy of the GNU Affero General Public License
 | 
						|
	along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | 
						|
 | 
						|
*/
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
#include <ortp/ortp.h>
 | 
						|
#include <osipparser2/sdp_message.h>
 | 
						|
 | 
						|
#include <GSMLogicalChannel.h>
 | 
						|
#include <GSMConfig.h>
 | 
						|
#include <ControlCommon.h>
 | 
						|
#include <TransactionTable.h>
 | 
						|
#include <SubscriberRegistry.h>
 | 
						|
 | 
						|
#include <Sockets.h>
 | 
						|
 | 
						|
#include "SIPUtility.h"
 | 
						|
#include "SIPInterface.h"
 | 
						|
 | 
						|
#include <Logger.h>
 | 
						|
 | 
						|
#undef WARNING
 | 
						|
 | 
						|
 | 
						|
 | 
						|
using namespace std;
 | 
						|
using namespace SIP;
 | 
						|
 | 
						|
using namespace GSM;
 | 
						|
using namespace Control;
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
// SIPMessageMap method definitions.
 | 
						|
 | 
						|
void SIPMessageMap::write(const std::string& call_id, osip_message_t * msg)
 | 
						|
{
 | 
						|
	LOG(DEBUG) << "call_id=" << call_id << " msg=" << msg;
 | 
						|
	OSIPMessageFIFO * fifo = mMap.readNoBlock(call_id);
 | 
						|
	if( fifo==NULL ) {
 | 
						|
		// FIXME -- If this write fails, send "call leg non-existent" response on SIP interface.
 | 
						|
		LOG(NOTICE) << "missing SIP FIFO "<<call_id;
 | 
						|
		throw SIPError();
 | 
						|
	}
 | 
						|
	LOG(DEBUG) << "write on fifo " << fifo;
 | 
						|
	fifo->write(msg);	
 | 
						|
}
 | 
						|
 | 
						|
osip_message_t * SIPMessageMap::read(const std::string& call_id, unsigned readTimeout)
 | 
						|
{ 
 | 
						|
	LOG(DEBUG) << "call_id=" << call_id;
 | 
						|
	OSIPMessageFIFO * fifo = mMap.readNoBlock(call_id);
 | 
						|
	if (!fifo) {
 | 
						|
		LOG(NOTICE) << "missing SIP FIFO "<<call_id;
 | 
						|
		throw SIPError();
 | 
						|
	}	
 | 
						|
	LOG(DEBUG) << "blocking on fifo " << fifo;
 | 
						|
	osip_message_t * msg =  fifo->read(readTimeout);	
 | 
						|
	if (!msg) throw SIPTimeout();
 | 
						|
	return msg;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
bool SIPMessageMap::add(const std::string& call_id, const struct sockaddr_in* returnAddress)
 | 
						|
{
 | 
						|
	OSIPMessageFIFO * fifo = new OSIPMessageFIFO(returnAddress);
 | 
						|
	mMap.write(call_id, fifo);
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
bool SIPMessageMap::remove(const std::string& call_id)
 | 
						|
{
 | 
						|
	OSIPMessageFIFO * fifo = mMap.readNoBlock(call_id);
 | 
						|
	if(fifo == NULL) return false;
 | 
						|
	mMap.remove(call_id);
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
// SIPInterface method definitions.
 | 
						|
 | 
						|
bool SIPInterface::addCall(const string &call_id)
 | 
						|
{
 | 
						|
	LOG(INFO) << "creating SIP message FIFO callID " << call_id;
 | 
						|
	return mSIPMap.add(call_id,mSIPSocket.source());
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
bool SIPInterface::removeCall(const string &call_id)
 | 
						|
{
 | 
						|
	LOG(INFO) << "removing SIP message FIFO callID " << call_id;
 | 
						|
	return mSIPMap.remove(call_id);
 | 
						|
}
 | 
						|
 | 
						|
int SIPInterface::fifoSize(const std::string& call_id )
 | 
						|
{ 
 | 
						|
	OSIPMessageFIFO * fifo = mSIPMap.map().read(call_id,0);
 | 
						|
	if(fifo==NULL) return -1;
 | 
						|
	return fifo->size();
 | 
						|
}	
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
void SIP::driveLoop( SIPInterface * si){
 | 
						|
	while (true) {
 | 
						|
		si->drive();
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void SIPInterface::start(){
 | 
						|
	// Start all the osip/ortp stuff. 
 | 
						|
	parser_init();
 | 
						|
	ortp_init();
 | 
						|
	ortp_scheduler_init();
 | 
						|
	// FIXME -- Can we coordinate this with the global logger?
 | 
						|
	//ortp_set_log_level_mask(ORTP_MESSAGE|ORTP_WARNING|ORTP_ERROR);
 | 
						|
	mDriveThread.start((void *(*)(void*))driveLoop,this );
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
void SIPInterface::write(const struct sockaddr_in* dest, osip_message_t *msg) 
 | 
						|
{
 | 
						|
	char * str;
 | 
						|
	size_t msgSize;
 | 
						|
	osip_message_to_str(msg, &str, &msgSize);
 | 
						|
	if (!str) {
 | 
						|
		LOG(ERR) << "osip_message_to_str produced a NULL pointer.";
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	//if it's any of these transactions, record it in the database
 | 
						|
	string name = osip_message_get_from(msg)->url->username;
 | 
						|
	if (msg->sip_method && 
 | 
						|
	    (!strncmp(msg->sip_method, "INVITE", 6) ||
 | 
						|
	     !strncmp(msg->sip_method, "REGISTER", 8) ||
 | 
						|
	     !strncmp(msg->sip_method, "MESSAGE", 7))) {
 | 
						|
		if (gSubscriberRegistry.imsiSet(name, "ipaddr", 
 | 
						|
						osip_message_get_from(msg)->url->host)){
 | 
						|
			LOG(INFO) << "SR ipaddr Update Problem";
 | 
						|
		}
 | 
						|
		if (gSubscriberRegistry.imsiSet(name, "port", 
 | 
						|
						gConfig.getStr("SIP.Local.Port"))){
 | 
						|
			LOG(INFO) << "SR port Update Problem";
 | 
						|
		}
 | 
						|
	}
 | 
						|
	char firstLine[100];
 | 
						|
	sscanf(str,"%100[^\n]",firstLine);
 | 
						|
	LOG(INFO) << "write " << firstLine;
 | 
						|
	LOG(DEBUG) << "write " << str;
 | 
						|
 | 
						|
	mSocketLock.lock();
 | 
						|
	mSIPSocket.send((const struct sockaddr*)dest,str);
 | 
						|
	mSocketLock.unlock();
 | 
						|
	free(str);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
void SIPInterface::drive() 
 | 
						|
{
 | 
						|
	// All inbound SIP messages go here for processing.
 | 
						|
 | 
						|
	LOG(DEBUG) << "blocking on socket";
 | 
						|
	int numRead = mSIPSocket.read(mReadBuffer);
 | 
						|
	if (numRead<0) {
 | 
						|
		LOG(ALERT) << "cannot read SIP socket.";
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	// FIXME -- Is this +1 offset correct?  Check it.
 | 
						|
	mReadBuffer[numRead] = '\0';
 | 
						|
 | 
						|
	// Get the proxy from the inbound message.
 | 
						|
#if 0
 | 
						|
	const struct sockaddr_in* sourceAddr = mSIPSocket.source();
 | 
						|
	char msgHost[256];
 | 
						|
	const char* msgHostRet = inet_ntop(AF_INET,&(sourceAddr->sin_addr),msgHost,255);
 | 
						|
	if (!msgHostRet) {
 | 
						|
		LOG(ALERT) << "cannot translate SIP source address for " << mReadBuffer;
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	unsigned msgPortNumber = sourceAddr->sin_port;
 | 
						|
	char msgPort[20];
 | 
						|
	sprintf(msgPort,"%u",msgPortNumber);
 | 
						|
	string proxy = string(msgHost) + string(":") + string(msgPort);
 | 
						|
#endif
 | 
						|
 | 
						|
	char firstLine[101];
 | 
						|
	sscanf(mReadBuffer,"%100[^\n]",firstLine);
 | 
						|
	LOG(INFO) << "read " << firstLine;
 | 
						|
	LOG(DEBUG) << "read " << mReadBuffer;
 | 
						|
 | 
						|
 | 
						|
	try {
 | 
						|
 | 
						|
		// Parse the mesage.
 | 
						|
		osip_message_t * msg;
 | 
						|
		int i = osip_message_init(&msg);
 | 
						|
		LOG(INFO) << "osip_message_init " << i;
 | 
						|
		int j = osip_message_parse(msg, mReadBuffer, strlen(mReadBuffer));
 | 
						|
		// seems like it ought to do something more than display an error,
 | 
						|
		// but it used to not even do that.
 | 
						|
		LOG(INFO) << "osip_message_parse " << j;
 | 
						|
 | 
						|
		if (msg->sip_method) LOG(DEBUG) << "read method " << msg->sip_method;
 | 
						|
 | 
						|
		// Must check if msg is an invite.
 | 
						|
		// if it is, handle appropriatly.
 | 
						|
		// FIXME -- Check return value in case this failed.
 | 
						|
		// FIXME -- If we support USSD via SIP, we will need to check the map first.
 | 
						|
		checkInvite(msg);
 | 
						|
 | 
						|
		// Multiplex out the received SIP message to active calls.
 | 
						|
 | 
						|
		// If we write to non-existent call_id.
 | 
						|
		// this is errant message so need to catch
 | 
						|
		// Internal error excatpion. and give nice
 | 
						|
		// message (instead of aborting)
 | 
						|
		// Don't free call_id_num.  It points into msg->call_id.
 | 
						|
		char * call_id_num = osip_call_id_get_number(msg->call_id);	
 | 
						|
		if( call_id_num == NULL ) {
 | 
						|
			LOG(WARNING) << "message with no call id";
 | 
						|
			throw SIPError();
 | 
						|
		}
 | 
						|
		LOG(DEBUG) << "got message " << msg << " with call id " << call_id_num << " and writing it to the map.";
 | 
						|
		string call_num(call_id_num);
 | 
						|
		// Don't free msg.  Whoever reads the FIFO will do that.
 | 
						|
		mSIPMap.write(call_num, msg);
 | 
						|
	}
 | 
						|
	catch(SIPException) {
 | 
						|
		LOG(NOTICE) << "discarded out-of-place SIP message: " << mReadBuffer;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
const char* extractIMSI(const osip_message_t *msg)
 | 
						|
{
 | 
						|
	// Get request username (IMSI) from invite. 
 | 
						|
	// Form of the name is IMSI<digits>, and it should always be 18 or 19 char.
 | 
						|
	const char * IMSI = msg->req_uri->username;
 | 
						|
	LOG(INFO) << msg->sip_method << " to "<< IMSI;
 | 
						|
	// IMSIs are 14 or 15 char + "IMSI" prefix
 | 
						|
	// FIXME -- We need better validity-checking.
 | 
						|
	unsigned namelen = strlen(IMSI);
 | 
						|
	if ((namelen>19)||(namelen<18)) {
 | 
						|
		LOG(WARNING) << "INVITE with malformed username \"" << IMSI << "\"";
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	// Skip first 4 char "IMSI".
 | 
						|
	return IMSI+4;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
const char* extractCallID(const osip_message_t* msg)
 | 
						|
{
 | 
						|
	if (!msg->call_id) return NULL;
 | 
						|
	return  osip_call_id_get_number(msg->call_id);	
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
bool SIPInterface::checkInvite( osip_message_t * msg)
 | 
						|
{
 | 
						|
	LOG(DEBUG);
 | 
						|
 | 
						|
	// This code dispatches new transactions coming from the network-side SIP interface.
 | 
						|
	// All transactions originating here are going to be mobile-terminated.
 | 
						|
	// Yes, this method is too long and needs to be broken up into smaller steps.
 | 
						|
 | 
						|
	// Is there even a method?
 | 
						|
	const char *method = msg->sip_method;
 | 
						|
	if (!method) return false;
 | 
						|
 | 
						|
	// Check for INVITE or MESSAGE methods.
 | 
						|
	// Check channel availability now, too.
 | 
						|
	GSM::ChannelType requiredChannel;
 | 
						|
	bool channelAvailable = false;
 | 
						|
	GSM::L3CMServiceType serviceType;
 | 
						|
	// pretty sure strings are garbage collected
 | 
						|
	string proxy = get_return_address(msg);
 | 
						|
	if (strcmp(method,"INVITE") == 0) {
 | 
						|
		// INVITE is for MTC.
 | 
						|
		// Set the required channel type to match the assignment style.
 | 
						|
		if (gConfig.defines("Control.VEA")) {
 | 
						|
			// Very early assignment.
 | 
						|
			requiredChannel = GSM::TCHFType;
 | 
						|
			channelAvailable = gBTS.TCHAvailable();
 | 
						|
		} else {
 | 
						|
			// Early assignment
 | 
						|
			requiredChannel = GSM::SDCCHType;
 | 
						|
			channelAvailable = gBTS.SDCCHAvailable() && gBTS.TCHAvailable();
 | 
						|
		}
 | 
						|
		serviceType = L3CMServiceType::MobileTerminatedCall;
 | 
						|
	}
 | 
						|
	else if (strcmp(method,"MESSAGE") == 0) {
 | 
						|
		// MESSAGE is for MTSMS.
 | 
						|
		requiredChannel = GSM::SDCCHType;
 | 
						|
		channelAvailable = gBTS.SDCCHAvailable();
 | 
						|
		serviceType = L3CMServiceType::MobileTerminatedShortMessage;
 | 
						|
	}
 | 
						|
	else {
 | 
						|
		// Not a method handled here.
 | 
						|
		LOG(DEBUG) << "non-initiating SIP method " << method;
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	// Get request username (IMSI) from invite. 
 | 
						|
	const char* IMSI = extractIMSI(msg);
 | 
						|
	if (!IMSI) {
 | 
						|
		// FIXME -- Send appropriate error (404) on SIP interface.
 | 
						|
		LOG(WARNING) << "Incoming INVITE/MESSAGE with no IMSI";
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	L3MobileIdentity mobileID(IMSI);
 | 
						|
 | 
						|
	// Get the SIP call ID.
 | 
						|
	const char * callIDNum = extractCallID(msg);	
 | 
						|
	if (!callIDNum) {
 | 
						|
		// FIXME -- Send appropriate error on SIP interface.
 | 
						|
		LOG(WARNING) << "Incoming INVITE/MESSAGE with no call ID";
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	// Find any active transaction for this IMSI with an assigned TCH or SDCCH.
 | 
						|
	GSM::LogicalChannel *chan = gTransactionTable.findChannel(mobileID);
 | 
						|
	bool userBusy = false;
 | 
						|
	if (chan) {
 | 
						|
		// If the type is TCH and the service is SMS, get the SACCH.
 | 
						|
		// Otherwise, for now, just say chan=NULL.
 | 
						|
		if (serviceType==L3CMServiceType::MobileTerminatedShortMessage && chan->type()==FACCHType) {
 | 
						|
			chan = chan->SACCH();
 | 
						|
		} else {
 | 
						|
			// FIXME -- This will change to support multiple transactions.
 | 
						|
			userBusy = (serviceType==L3CMServiceType::MobileTerminatedCall) && !(chan->recyclable());
 | 
						|
			chan = NULL;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	// Check SIP map.  Repeated entry?  Page again.
 | 
						|
	if (mSIPMap.map().readNoBlock(callIDNum) != NULL) { 
 | 
						|
		TransactionEntry* transaction= gTransactionTable.find(mobileID,callIDNum);
 | 
						|
		// There's a FIFO but no trasnaction record?
 | 
						|
		if (!transaction) {
 | 
						|
			LOG(WARNING) << "repeated INVITE/MESSAGE with no transaction record";
 | 
						|
			// Delete the bogus FIFO.
 | 
						|
			mSIPMap.remove(callIDNum);
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
		//if this is not the saved invite, it's a RE-invite. Respond saying we don't support it. 
 | 
						|
		if (!transaction->sameINVITE(msg)){
 | 
						|
			/* don't cancel the call */
 | 
						|
			LOG(CRIT) << "got reinvite. transaction: " << *transaction << " SIP re-INVITE: " << msg;
 | 
						|
			transaction->MODSendERROR(msg, 488, "Not Acceptable Here", false);
 | 
						|
			/* I think we'd need to create a new transaction for this ack. Right now, just assume the ack makes it back. 
 | 
						|
			   if not, we'll hear another INVITE */
 | 
						|
			//transaction->MODWaitForERRORACK(false); //don't cancel the call
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
		// There is transaction already.  Send trying.
 | 
						|
		transaction->MTCSendTrying();
 | 
						|
		// And if no channel is established yet, page again.
 | 
						|
		if (!chan) {
 | 
						|
			LOG(INFO) << "repeated SIP INVITE/MESSAGE, repaging for transaction " << *transaction; 
 | 
						|
			gBTS.pager().addID(mobileID,requiredChannel,*transaction);
 | 
						|
		}
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	// So we will need a new channel.
 | 
						|
	// Check gBTS for channel availability.
 | 
						|
	if (!chan && !channelAvailable) {
 | 
						|
		// FIXME -- Send 503 "Service Unavailable" response on SIP interface.
 | 
						|
		// Don't forget the retry-after header.
 | 
						|
		LOG(NOTICE) << "MTC CONGESTION, no " << requiredChannel << " availble for assignment";
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	if (chan)  { LOG(INFO) << "using existing channel " << chan->descriptiveString(); }
 | 
						|
	else { LOG(INFO) << "set up MTC paging for channel=" << requiredChannel; }
 | 
						|
 | 
						|
 | 
						|
	// Add an entry to the SIP Map to route inbound SIP messages.
 | 
						|
	addCall(callIDNum);
 | 
						|
	LOG(DEBUG) << "callIDNum " << callIDNum << " IMSI " << IMSI;
 | 
						|
 | 
						|
	// Get the caller ID if it's available.
 | 
						|
	const char *callerID = "";
 | 
						|
	const char *callerHost = "";
 | 
						|
	osip_from_t *from = osip_message_get_from(msg);
 | 
						|
	if (from) {
 | 
						|
		osip_uri_t* url = osip_contact_get_url(from);
 | 
						|
		if (url) {
 | 
						|
			if (url->username) callerID = url->username;
 | 
						|
			if (url->host) callerHost = url->host;
 | 
						|
		}
 | 
						|
	} else {
 | 
						|
		LOG(NOTICE) << "INVITE with no From: username for " << mobileID;
 | 
						|
	}
 | 
						|
	LOG(DEBUG) << "callerID " << callerID << "@" << callerHost;
 | 
						|
 | 
						|
 | 
						|
	// Build the transaction table entry.
 | 
						|
	// This constructor sets TI automatically for an MT transaction.
 | 
						|
	TransactionEntry *transaction = new TransactionEntry(proxy.c_str(),mobileID,chan,serviceType,callerID);
 | 
						|
	// FIXME -- These parameters should be arguments to the constructor above.
 | 
						|
	transaction->SIPUser(callIDNum,IMSI,callerID,callerHost);
 | 
						|
	transaction->saveINVITE(msg,false);
 | 
						|
	// Tell the sender we are trying.
 | 
						|
	transaction->MTCSendTrying();
 | 
						|
 | 
						|
	// SMS?  Get the text message body to deliver.
 | 
						|
	if (serviceType == L3CMServiceType::MobileTerminatedShortMessage) {
 | 
						|
		osip_body_t *body;
 | 
						|
		osip_content_type_t *contentType;
 | 
						|
		osip_message_get_body(msg,0,&body);
 | 
						|
		contentType = osip_message_get_content_type(msg);
 | 
						|
		const char *text = NULL;
 | 
						|
		char *type = NULL;
 | 
						|
		if (body) text = body->body;
 | 
						|
		if (text) transaction->message(text, body->length);
 | 
						|
		else LOG(NOTICE) << "MTSMS incoming MESSAGE method with no message body for " << mobileID;
 | 
						|
		/* Ok, so osip does some funny stuff here. The MIME type is split into type and subType.
 | 
						|
			Basically, text/plain becomes type=text, subType=plain. We need to put those together...
 | 
						|
		*/
 | 
						|
		if (contentType) {
 | 
						|
			type = (char *)malloc(strlen(contentType->type)+strlen(contentType->subtype)+2);
 | 
						|
		}
 | 
						|
		if (type) {
 | 
						|
			strcpy(type,contentType->type);
 | 
						|
			strcat(type,"/");
 | 
						|
			strcat(type,contentType->subtype);
 | 
						|
			transaction->messageType(type);
 | 
						|
			free(type);
 | 
						|
		}
 | 
						|
		else LOG(NOTICE) << "MTSMS incoming MESSAGE method with no content type (or memory error) for " << mobileID;
 | 
						|
	}
 | 
						|
 | 
						|
	LOG(INFO) << "MTC MTSMS make transaction and add to transaction table: "<< *transaction;
 | 
						|
	gTransactionTable.add(transaction); 
 | 
						|
 | 
						|
	if (userBusy) {
 | 
						|
        	transaction->MODSendERROR(msg, 486, "Busy Here", false);
 | 
						|
        	return true;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	// If there's an existing channel, skip the paging step.
 | 
						|
	if (!chan) {
 | 
						|
		// Add to paging list.
 | 
						|
		LOG(DEBUG) << "MTC MTSMS new SIP invite, initial paging for mobile ID " << mobileID;
 | 
						|
		gBTS.pager().addID(mobileID,requiredChannel,*transaction);	
 | 
						|
	} else {
 | 
						|
		// Add a transaction to an existing channel.
 | 
						|
		chan->addTransaction(transaction);
 | 
						|
		// FIXME -- We need to write something into the channel to trigger the new transaction.
 | 
						|
		// We need to send a message into the chan's dispatch loop,
 | 
						|
		// becasue we can't block this thread to run the transaction.
 | 
						|
	}
 | 
						|
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
// vim: ts=4 sw=4
 |