mirror of
				https://github.com/RangeNetworks/openbts.git
				synced 2025-10-31 03:43:33 +00:00 
			
		
		
		
	git-svn-id: http://wush.net/svn/range/software/public/openbts/trunk@2242 19bc5d8c-e614-43d4-8b26-e1612bc8e597
		
			
				
	
	
		
			193 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			193 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /**@file Idle-mode dispatcher for dedicated control channels. */
 | |
| 
 | |
| /*
 | |
| * Copyright 2008, 2009, 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 "ControlCommon.h"
 | |
| #include "TransactionTable.h"
 | |
| #include "RadioResource.h"
 | |
| #include "MobilityManagement.h"
 | |
| #include <GSMLogicalChannel.h>
 | |
| #include <GSML3MMMessages.h>
 | |
| #include <GSML3RRMessages.h>
 | |
| #include <SIPUtility.h>
 | |
| #include <SIPInterface.h>
 | |
| 
 | |
| #include <Logger.h>
 | |
| #undef WARNING
 | |
| 
 | |
| using namespace std;
 | |
| using namespace GSM;
 | |
| using namespace Control;
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| /**
 | |
| 	Dispatch the appropriate controller for a Mobility Management message.
 | |
| 	@param req A pointer to the initial message.
 | |
| 	@param DCCH A pointer to the logical channel for the transaction.
 | |
| */
 | |
| void DCCHDispatchMM(const L3MMMessage* req, LogicalChannel *DCCH)
 | |
| {
 | |
| 	assert(req);
 | |
| 	L3MMMessage::MessageType MTI = (L3MMMessage::MessageType)req->MTI();
 | |
| 	switch (MTI) {
 | |
| 		case L3MMMessage::LocationUpdatingRequest:
 | |
| 			LocationUpdatingController(dynamic_cast<const L3LocationUpdatingRequest*>(req),DCCH);
 | |
| 			break;
 | |
| 		case L3MMMessage::IMSIDetachIndication:
 | |
| 			IMSIDetachController(dynamic_cast<const L3IMSIDetachIndication*>(req),DCCH);
 | |
| 			break;
 | |
| 		case L3MMMessage::CMServiceRequest:
 | |
| 			CMServiceResponder(dynamic_cast<const L3CMServiceRequest*>(req),DCCH);
 | |
| 			break;
 | |
| 		default:
 | |
| 			LOG(NOTICE) << "unhandled MM message " << MTI << " on " << *DCCH;
 | |
| 			throw UnsupportedMessage();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
| 	Dispatch the appropriate controller for a Radio Resource message.
 | |
| 	@param req A pointer to the initial message.
 | |
| 	@param DCCH A pointer to the logical channel for the transaction.
 | |
| */
 | |
| void DCCHDispatchRR(const L3RRMessage* req, LogicalChannel *DCCH)
 | |
| {
 | |
| 	LOG(DEBUG) << "checking MTI"<< (L3RRMessage::MessageType)req->MTI();
 | |
| 
 | |
| 	// TODO SMS -- This needs to handle SACCH Measurement Reports.
 | |
| 
 | |
| 	assert(req);
 | |
| 	L3RRMessage::MessageType MTI = (L3RRMessage::MessageType)req->MTI();
 | |
| 	switch (MTI) {
 | |
| 		case L3RRMessage::PagingResponse:
 | |
| 			PagingResponseHandler(dynamic_cast<const L3PagingResponse*>(req),DCCH);
 | |
| 			break;
 | |
| 		case L3RRMessage::AssignmentComplete:
 | |
| 			AssignmentCompleteHandler(dynamic_cast<const L3AssignmentComplete*>(req),
 | |
| 										dynamic_cast<TCHFACCHLogicalChannel*>(DCCH));
 | |
| 			break;
 | |
| 		default:
 | |
| 			LOG(NOTICE) << "unhandled RR message " << MTI << " on " << *DCCH;
 | |
| 			throw UnsupportedMessage();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| void DCCHDispatchMessage(const L3Message* msg, LogicalChannel* DCCH)
 | |
| {
 | |
| 	// Each protocol has it's own sub-dispatcher.
 | |
| 	switch (msg->PD()) {
 | |
| 		case L3MobilityManagementPD:
 | |
| 			DCCHDispatchMM(dynamic_cast<const L3MMMessage*>(msg),DCCH);
 | |
| 			break;
 | |
| 		case L3RadioResourcePD:
 | |
| 			DCCHDispatchRR(dynamic_cast<const L3RRMessage*>(msg),DCCH);
 | |
| 			break;
 | |
| 		default:
 | |
| 			LOG(NOTICE) << "unhandled protocol " << msg->PD() << " on " << *DCCH;
 | |
| 			throw UnsupportedMessage();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| /** Example of a closed-loop, persistent-thread control function for the DCCH. */
 | |
| void Control::DCCHDispatcher(LogicalChannel *DCCH)
 | |
| {
 | |
| 	while (1) {
 | |
| 		try {
 | |
| 			// Wait for a transaction to start.
 | |
| 			LOG(DEBUG) << "waiting for " << *DCCH << " ESTABLISH";
 | |
| 			DCCH->waitForPrimitive(ESTABLISH);
 | |
| 			// Pull the first message and dispatch a new transaction.
 | |
| 			const L3Message *message = getMessage(DCCH);
 | |
| 			LOG(DEBUG) << *DCCH << " received " << *message;
 | |
| 			DCCHDispatchMessage(message,DCCH);
 | |
| 			delete message;
 | |
| 		}
 | |
| 
 | |
| 		// Catch the various error cases.
 | |
| 
 | |
| 		catch (ChannelReadTimeout except) {
 | |
| 			LOG(NOTICE) << "ChannelReadTimeout";
 | |
| 			// Cause 0x03 means "abnormal release, timer expired".
 | |
| 			DCCH->send(L3ChannelRelease(0x03));
 | |
| 			gTransactionTable.remove(except.transactionID());
 | |
| 		}
 | |
| 		catch (UnexpectedPrimitive except) {
 | |
| 			LOG(NOTICE) << "UnexpectedPrimitive";
 | |
| 			// Cause 0x62 means "message type not not compatible with protocol state".
 | |
| 			DCCH->send(L3ChannelRelease(0x62));
 | |
| 			if (except.transactionID()) gTransactionTable.remove(except.transactionID());
 | |
| 		}
 | |
| 		catch (UnexpectedMessage except) {
 | |
| 			LOG(NOTICE) << "UnexpectedMessage";
 | |
| 			// Cause 0x62 means "message type not not compatible with protocol state".
 | |
| 			DCCH->send(L3ChannelRelease(0x62));
 | |
| 			if (except.transactionID()) gTransactionTable.remove(except.transactionID());
 | |
| 		}
 | |
| 		catch (UnsupportedMessage except) {
 | |
| 			LOG(NOTICE) << "UnsupportedMessage";
 | |
| 			// Cause 0x61 means "message type not implemented".
 | |
| 			DCCH->send(L3ChannelRelease(0x61));
 | |
| 			if (except.transactionID()) gTransactionTable.remove(except.transactionID());
 | |
| 		}
 | |
| 		catch (Q931TimerExpired except) {
 | |
| 			LOG(NOTICE) << "Q.931 T3xx timer expired";
 | |
| 			// Cause 0x03 means "abnormal release, timer expired".
 | |
| 			// TODO -- Send diagnostics.
 | |
| 			DCCH->send(L3ChannelRelease(0x03));
 | |
| 			if (except.transactionID()) gTransactionTable.remove(except.transactionID());
 | |
| 		}
 | |
| 		catch (SIP::SIPTimeout except) {
 | |
| 			// FIXME -- The transaction ID should be an argument here.
 | |
| 			LOG(WARNING) << "Uncaught SIPTimeout, will leave a stray transcation";
 | |
| 			// Cause 0x03 means "abnormal release, timer expired".
 | |
| 			DCCH->send(L3ChannelRelease(0x03));
 | |
| 			if (except.transactionID()) gTransactionTable.remove(except.transactionID());
 | |
| 		}
 | |
| 		catch (SIP::SIPError except) {
 | |
| 			// FIXME -- The transaction ID should be an argument here.
 | |
| 			LOG(WARNING) << "Uncaught SIPError, will leave a stray transcation";
 | |
| 			// Cause 0x01 means "abnormal release, unspecified".
 | |
| 			DCCH->send(L3ChannelRelease(0x01));
 | |
| 			if (except.transactionID()) gTransactionTable.remove(except.transactionID());
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| // vim: ts=4 sw=4
 |