mirror of
https://github.com/RangeNetworks/openbts.git
synced 2025-10-29 02:53:35 +00:00
merged in commercial openbts
This commit is contained in:
@@ -18,17 +18,23 @@
|
||||
*/
|
||||
|
||||
|
||||
#define LOG_GROUP LogGroup::Control
|
||||
|
||||
|
||||
#include "ControlCommon.h"
|
||||
#include "TransactionTable.h"
|
||||
//#include "L3CallControl.h"
|
||||
#include "L3MobilityManagement.h"
|
||||
#include "L3StateMachine.h"
|
||||
#include "L3LogicalChannel.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 "MobilityManagement.h"
|
||||
//#include <GSMLogicalChannel.h>
|
||||
//#include <GSML3Message.h>
|
||||
//#include <GSML3MMMessages.h>
|
||||
//#include <GSML3RRMessages.h>
|
||||
//#include <SIPUtility.h>
|
||||
//#include <SIPInterface.h>
|
||||
|
||||
#include <Logger.h>
|
||||
#undef WARNING
|
||||
@@ -42,111 +48,25 @@ 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();
|
||||
|
||||
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. */
|
||||
// (pat) DCCH is a TCHFACCHLogicalChannel or SDCCHLogicalChannel
|
||||
void Control::DCCHDispatcher(LogicalChannel *DCCH)
|
||||
void Control::DCCHDispatcher(L3LogicalChannel *DCCH)
|
||||
{
|
||||
while (1) {
|
||||
// This 'try' is redundant, but we are ultra-cautious here since a mistake means a crash.
|
||||
try {
|
||||
// Wait for a transaction to start.
|
||||
LOG(DEBUG) << "waiting for " << *DCCH << " ESTABLISH or HANDOVER_ACCESS";
|
||||
L3Frame *frame = DCCH->waitForEstablishOrHandover();
|
||||
LOG(DEBUG) << *DCCH << " received " << *frame;
|
||||
gResetWatchdog();
|
||||
Primitive prim = frame->primitive();
|
||||
delete frame;
|
||||
LOG(DEBUG) << "received primtive " << prim;
|
||||
switch (prim) {
|
||||
case ESTABLISH: {
|
||||
// Pull the first message and dispatch a new transaction.
|
||||
gReports.incr("OpenBTS.GSM.RR.ChannelSiezed");
|
||||
const L3Message *message = getMessage(DCCH);
|
||||
LOG(INFO) << *DCCH << " received establishing messaage " << *message;
|
||||
DCCHDispatchMessage(message,DCCH);
|
||||
delete message;
|
||||
break;
|
||||
}
|
||||
case HANDOVER_ACCESS: {
|
||||
ProcessHandoverAccess(dynamic_cast<GSM::TCHFACCHLogicalChannel*>(DCCH));
|
||||
break;
|
||||
}
|
||||
default: assert(0);
|
||||
}
|
||||
L3DCCHLoop(DCCH,frame); // This will not return until the channel is released.
|
||||
}
|
||||
catch (...) {
|
||||
LOG(ERR) << "channel killed by unexpected exception ";
|
||||
}
|
||||
|
||||
#if 0
|
||||
// Catch the various error cases.
|
||||
|
||||
catch (RemovedTransaction except) {
|
||||
@@ -155,48 +75,49 @@ void Control::DCCHDispatcher(LogicalChannel *DCCH)
|
||||
catch (ChannelReadTimeout except) {
|
||||
LOG(NOTICE) << "ChannelReadTimeout";
|
||||
// Cause 0x03 means "abnormal release, timer expired".
|
||||
DCCH->send(L3ChannelRelease(0x03));
|
||||
DCCH->l2sendm(L3ChannelRelease((RRCause)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));
|
||||
DCCH->l2sendm(L3ChannelRelease((RRCause)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));
|
||||
DCCH->l2sendm(L3ChannelRelease((RRCause)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));
|
||||
DCCH->l2sendm(L3ChannelRelease((RRCause)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));
|
||||
DCCH->l2sendm(L3ChannelRelease((RRCause)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));
|
||||
DCCH->l2sendm(L3ChannelRelease((RRCause)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));
|
||||
DCCH->l2sendm(L3ChannelRelease((RRCause)0x01));
|
||||
if (except.transactionID()) gTransactionTable.remove(except.transactionID());
|
||||
}
|
||||
#endif
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user