OpenBTS-UMTS/UMTS/URRCMessages.cpp

2060 lines
94 KiB
C++

/*
* OpenBTS provides an open source alternative to legacy telco protocols and
* traditionally complex, proprietary hardware systems.
*
* Copyright 2014 Range Networks, Inc.
*
* This software is distributed under the terms of the GNU Affero General
* Public License version 3. See the COPYING and NOTICE files in the main
* directory for licensing information.
*
* This use of this software may be subject to additional restrictions.
* See the LEGAL file in the main directory for details.
*/
#include "UMTSTransfer.h"
#include "URRCMessages.h"
#include "MACEngine.h"
#include "AsnHelper.h"
#include "Logger.h"
#include "SgsnExport.h"
#include "URRC.h"
#include "UMTSLogicalChannel.h"
//#include "asn_system.h" included from AsnHelper.h
namespace ASN {
//#include "BIT_STRING.h"
#include "UL-CCCH-Message.h"
#include "DL-CCCH-Message.h"
#include "UL-DCCH-Message.h"
#include "DL-DCCH-Message.h"
#include "InitialUE-Identity.h"
#define PAT_SAMSUNG_TEST 1 // Try to get the samsung galaxy to accept this message.
#include "asn_SEQUENCE_OF.h"
//#include "RRCConnectionRequest.h"
//#include "RRCConnectionSetup.h"
};
#define CASENAME(x) case x: return #x;
#define CASEASNCOMMENT(foo) case ASN::foo: return #foo;
using namespace SGSN;
namespace UMTS {
const std::string descrRrcConnectionSetup("RRC_Connection_Setup_Message");
const std::string descrRrcConnectionRelease("RRC_Connection_Release_Message");
const std::string descrRadioBearerSetup("RRC Radio Bearer Setup Message");
const std::string descrRadioBearerRelease("RRC Radio Bearer Release Message");
const std::string descrCellUpdateConfirm("RRC Cell Update Confirm Message");
const std::string descrSecurityModeCommand("RRC Security Mode Command");
typedef unsigned char uchar;
static const char *asnUlDcchMsg2Name(ASN::UL_DCCH_MessageType_PR mtype)
{
switch (mtype) {
CASEASNCOMMENT(UL_DCCH_MessageType_PR_activeSetUpdateComplete)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_activeSetUpdateFailure)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_cellChangeOrderFromUTRANFailure)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_counterCheckResponse)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_handoverToUTRANComplete)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_initialDirectTransfer)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_handoverFromUTRANFailure)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_measurementControlFailure)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_measurementReport)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_physicalChannelReconfigurationComplete)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_physicalChannelReconfigurationFailure)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_radioBearerReconfigurationComplete)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_radioBearerReconfigurationFailure)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_radioBearerReleaseComplete)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_radioBearerReleaseFailure)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_radioBearerSetupComplete)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_radioBearerSetupFailure)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_rrcConnectionReleaseComplete)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_rrcConnectionSetupComplete)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_rrcStatus)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_securityModeComplete)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_securityModeFailure)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_signallingConnectionReleaseIndication)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_transportChannelReconfigurationComplete)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_transportChannelReconfigurationFailure)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_transportFormatCombinationControlFailure)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_ueCapabilityInformation)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_uplinkDirectTransfer)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_utranMobilityInformationConfirm)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_utranMobilityInformationFailure)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_mbmsModificationRequest)
CASEASNCOMMENT(UL_DCCH_MessageType_PR_spare1)
default: return "UL_DCCH_unrecognized_message_type";
}
}
static const char *asnUlCcchMsg2Name(ASN::UL_CCCH_MessageType_PR type)
{
switch (type) {
CASEASNCOMMENT(UL_CCCH_MessageType_PR_cellUpdate)
CASEASNCOMMENT(UL_CCCH_MessageType_PR_rrcConnectionRequest)
CASEASNCOMMENT(UL_CCCH_MessageType_PR_uraUpdate)
default:
return "UL_CCCH_unrecognized_message_type";
}
}
static string getRBFailureCause(ASN::FailureCauseWithProtErr_t *failureCause)
{
unsigned failcode = failureCause->present;
string msg;
switch (failcode) {
case ASN::FailureCauseWithProtErr_PR_configurationUnsupported:
msg = "configurationUnsupported"; break;
case ASN::FailureCauseWithProtErr_PR_physicalChannelFailure:
msg = "physicalChannelASN::Failure"; break;
case ASN::FailureCauseWithProtErr_PR_incompatibleSimultaneousReconfiguration:
msg = "incompatibleSimultaneousReconfiguration"; break;
case ASN::FailureCauseWithProtErr_PR_compressedModeRuntimeError:
msg = "compressedModeRuntimeError"; break;
case ASN::FailureCauseWithProtErr_PR_protocolError:
msg = "protocolError";
// The protocolError choice has additional info, but since our spec has only one
// possible value for that info, it is of little interest so ignore it.
break;
case ASN::FailureCauseWithProtErr_PR_cellUpdateOccurred:
msg = "cellUpdateOccurred"; break;
case ASN::FailureCauseWithProtErr_PR_invalidConfiguration:
msg = "invalidConfiguration"; break;
case ASN::FailureCauseWithProtErr_PR_configurationIncomplete:
msg = "configurationIncomplete"; break;
case ASN::FailureCauseWithProtErr_PR_unsupportedMeasurement:
msg = "unsupportedMeasurement"; break;
case ASN::FailureCauseWithProtErr_PR_mbmsSessionAlreadyReceivedCorrectly:
msg ="mbmsSessionAlreadyReceivedCorrectly"; break;
case ASN::FailureCauseWithProtErr_PR_lowerPriorityMBMSService:
msg = "lowerPriorityMBMSService"; break;
default:
msg = "unknown type"; break;
}
return msg;
}
ASN::RRC_StateIndicator UEState2Asn(UEState state)
{
switch (state) {
case stIdleMode:
LOG(ERR) << "UE is unexpectedly in idle mode";
// Assume CELL_FACH and fall through.
return ASN::RRC_StateIndicator_cell_FACH;
case stCELL_FACH: return ASN::RRC_StateIndicator_cell_FACH;
case stCELL_DCH: return ASN::RRC_StateIndicator_cell_DCH;
case stCELL_PCH: return ASN::RRC_StateIndicator_cell_PCH;
case stURA_PCH: return ASN::RRC_StateIndicator_ura_PCH;
default: assert(0);
}
}
// Run the Integrity Protection Algorithm and ASN encode the message.
// Return result in &result.
static bool encodeDcchMsg(UEInfo *uep, RbId rbid, ASN::DL_DCCH_Message_t *msg, ByteVector &result, string descr)
{
if (uep->integrity.isStarted()) {
// Step 1: Set the integrity check info to the values for encoding specified in 10.3.3.16.
ASN::IntegrityCheckInfo *ici = RN_CALLOC(ASN::IntegrityCheckInfo);
msg->integrityCheckInfo = ici;
ici->messageAuthenticationCode = allocAsnBIT_STRING(32);
// The RBID goes in the 5 LSB of the 32-bit string that will
// later hold the MAC-I [Message Authentication Code].
AsnBitString2BVTemp(ici->messageAuthenticationCode).setField(32-5,rbid-0*1,5);
ici->rrc_MessageSequenceNumber = 0; // 10.3.3.16 of 25.331;
// Step 2: Encode the message
if (!uperEncodeToBV(&ASN::asn_DEF_DL_DCCH_Message,msg,result,descr)) {return false;}
// After encoding it is safe to advance the COUNT-I, ie, this message advances
// the RRC SN [sequence number.]
//uep->integrity.advanceDlRrcSn(rbid);
// Step 3: Run the message through the Integrity Protection Algorithm.
uint32_t maci = uep->integrity.runF9(rbid,1,result);
// After encoding it is safe to advance the COUNT-I, ie, this message advances
// the RRC SN [sequence number.]
ici->rrc_MessageSequenceNumber = uep->integrity.getDlRrcSn(rbid); // 10.3.3.16 of 25.331
uep->integrity.advanceDlRrcSn(rbid);
// Step 4: Set the Message Authentication Code in the message.
AsnBitString2BVTemp(ici->messageAuthenticationCode).setField(0,maci,32);
}
// Encode the message (again), completely oblivious to cpu cycles consumed.
if (!uperEncodeToBV(&ASN::asn_DEF_DL_DCCH_Message,msg,result,descr)) {return false;}
std::string comment = format("DL_DCCH %s message size=%d",descr.c_str(),result.size());
asnLogMsg(rbid, &ASN::asn_DEF_DL_DCCH_Message, msg,comment.c_str(),uep);
//if (gConfig.getNum("UMTS.Debug.Messages")) {
// asn_fprint(stdout,&ASN::asn_DEF_DL_DCCH_Message, msg);
// std::string readable = asn2string(&ASN::asn_DEF_DL_DCCH_Message, msg);
// fprintf(stdout,"Encoded message size=%d bytes\n%s",result.size());
// fflush(stdout);
//}
return true;
}
// Return TRUE on success.
static bool encodeCcchMsg(ASN::DL_CCCH_Message_t *msg, ByteVector &result,
string descr, // Description for the log
UEInfo *uep, // UE for the log, or NULL if none.
uint32_t urnti // URNTI for the log, unneeded if uep is non-null.
)
{
// We can just return: uperEncodeToBV printed a message on failure.
bool stat = uperEncodeToBV(&ASN::asn_DEF_DL_CCCH_Message,msg,result,descr);
if (stat) {
string comment = format("DL_CCCH %s message size=%d",descr.c_str(),result.size());
asnLogMsg(0, &ASN::asn_DEF_DL_CCCH_Message, msg,comment.c_str(),uep,urnti);
}
//if (gConfig.getNum("UMTS.Debug.Messages")) {
// asn_fprint(stdout,&ASN::asn_DEF_DL_CCCH_Message, msg);
// fprintf(stdout,"Encoded message size=%d bytes\n",result.size());
// fflush(stdout);
//}
return stat;
}
// Same as RB_InformationSetup but without PDCP info.
// The list we put these things in may be either SRB_InformationSetupList or SRB_InformationSetupList2,
// which are 100% identical, but nevertheless the result list type must be void**
// 25.331 10.3.4.24
static void toAsnSRB_InformationSetupList(RrcMasterChConfig *masterConfig, void*srblist)
{
// Set up SRBs.
for (unsigned rbid = SRB1; rbid <= SRB3; rbid++) {
RBInfo *rb = masterConfig->getRB(rbid);
assert(rb->valid());
ASN::SRB_InformationSetup *srbie = RN_CALLOC(ASN::SRB_InformationSetup);
srbie->rb_Identity = RN_CALLOC(long);
*srbie->rb_Identity = rbid;
srbie->rlc_InfoChoice.present = ASN::RLC_InfoChoice_PR_rlc_Info;
rb->toAsnRLC_Info(&srbie->rlc_InfoChoice.choice.rlc_Info);
// This assumes uplink and downlink are configured with identical TrCh.
// They dont have to be.
TrChId tcid = rb->mTrChAssigned;
defaultRbMappingInfoToAsn(masterConfig->getUlTrChInfo(tcid),
masterConfig->getDlTrChInfo(tcid),rbid,&srbie->rb_MappingInfo);
ASN_SEQUENCE_ADD(srblist,srbie);
}
}
// This is almost the same as above, but allows you to change
// only the RBMappinginfo, not other RLC programming.
// We can use it to switch the SRBs between CELL_DCH and CELL_FACH mode,
// because it allows us to specify a new TrCh, which implies a new TFS,
// which allows the new rlcs to resize to match the new TFS.
static ASN::RB_InformationAffectedList *toAsnRB_InformationAffectedList(
RrcMasterChConfig *masterConfig)
{
ASN::RB_InformationAffectedList *result = RN_CALLOC(ASN::RB_InformationAffectedList);
// Set up SRBs.
for (unsigned rbid = SRB1; rbid <= SRB3; rbid++) {
RBInfo *rb = masterConfig->getRB(rbid);
assert(rb->valid());
ASN::RB_InformationAffected *srbie = RN_CALLOC(ASN::RB_InformationAffected);
srbie->rb_Identity = rbid;
// This assumes uplink and downlink are configured with identical TrCh.
// They dont have to be.
TrChId tcid = rb->mTrChAssigned;
defaultRbMappingInfoToAsn(masterConfig->getUlTrChInfo(tcid),
masterConfig->getDlTrChInfo(tcid),rbid,&srbie->rb_MappingInfo);
ASN_SEQUENCE_ADD(&result->list,srbie);
}
return result;
}
static void toAsnRAB_Identity(int rbid,ASN::RAB_Identity_t *rabp)
{
// 10.3.1.14: RAB Identity "This information uniquely identifies a RAB within a CN domain."
// TODO: There are different kinds for GSM-MAP or ANSI-41, but I dont think it matters
// because we dont use it and the phone shouldnt care, although it may need to be unique,
// which we can insure simply by using the rbid.
rabp->present = ASN::RAB_Identity_PR_gsm_MAP_RAB_Identity;
setAsnBIT_STRING(&rabp->choice.gsm_MAP_RAB_Identity,(uint8_t*)calloc(1,1),8);
AsnBitString2BVTemp(&rabp->choice.gsm_MAP_RAB_Identity).setField(0,rbid,8);
}
// 3GPP 25.331 10.3.4.8
// ASN RAB_Info_t rab_Info;
void toAsnRAB_Info(RBInfo *rb,ASN::RAB_Info_t *rabInfoIE)
{
// RAB_Identity_t rab_Identity;
toAsnRAB_Identity(rb->mRbId,&rabInfoIE->rab_Identity);
// CN_DomainIdentity_t cn_DomainIdentity; // cs or ps?
bool csdomain = rb->isCsDomain();
rabInfoIE->cn_DomainIdentity = toAsnEnumerated(
csdomain ? ASN::CN_DomainIdentity_cs_domain : ASN::CN_DomainIdentity_ps_domain);
// NAS_Synchronisation_Indicator_t *nas_Synchronisation_Indicator /* OPTIONAL */;
// Re_EstablishmentTimer_t re_EstablishmentTimer;
// TODO: Which should we use here? Choice is T314 or T315
// 25.331 13.1 implies T314 is for CS and T315 is for PS, so why are they asking us?
// Also says see 8.3.1.13 and 8.3.1.14
rabInfoIE->re_EstablishmentTimer = toAsnEnumerated(
csdomain ? ASN::Re_EstablishmentTimer_useT314 : ASN::Re_EstablishmentTimer_useT315);
// Finished with RAB_Info_t
}
// 3GPP 10.3.4.20
// ASN RB_InformationSetup
static ASN::RB_InformationSetup *toAsnRB_InformationSetup(RrcMasterChConfig *masterConfig, RBInfo *rb)
{
assert(rb->valid());
int rbid = rb->mRbId;
ASN::RB_InformationSetup *rbie = RN_CALLOC(ASN::RB_InformationSetup);
// RB_Identity_t rb_Identity;
rbie->rb_Identity = rbid;
// struct PDCP_Info *pdcp_Info /* OPTIONAL */;
// We dont use PDCP. So why is it here at all? Maybe it is mandatory for non-signalling RBs?
rbie->pdcp_Info = RN_CALLOC(ASN::PDCP_Info);
rbie->pdcp_Info->losslessSRNS_RelocSupport = RN_CALLOC(ASN::LosslessSRNS_RelocSupport);
rbie->pdcp_Info->losslessSRNS_RelocSupport->present = ASN::LosslessSRNS_RelocSupport_PR_notSupported;
//rbie->pdcpInfo->mLosslessSRNS_RelocSupport->choice = ASN::NULL;
rbie->pdcp_Info->pdcp_PDU_Header = toAsnEnumerated(ASN::PDCP_PDU_Header_absent);
// RLC_InfoChoice_t rlc_InfoChoice;
rbie->rlc_InfoChoice.present = ASN::RLC_InfoChoice_PR_rlc_Info;
rb->toAsnRLC_Info(&rbie->rlc_InfoChoice.choice.rlc_Info);
// RB_MappingInfo_t rb_MappingInfo;
TrChId tcid = rb->mTrChAssigned; // The TrCh carrying this rb.
defaultRbMappingInfoToAsn(masterConfig->getUlTrChInfo(tcid),
masterConfig->getDlTrChInfo(tcid),rbid,&rbie->rb_MappingInfo);
return rbie;
}
// 3GPP 25.331 10.3.4.10
// ASN struct RAB_InformationSetup
// For a CS connection we use three consecutive rbids.
static ASN::RAB_InformationSetup *toAsnRAB_InformationSetup(
RrcMasterChConfig *masterConfig, int rbid,
int *pNumRBperRAB) // Return the number of RBs in this RAB. 1 for PS and 3 for CS.
{
ASN::RAB_InformationSetup *result = RN_CALLOC(ASN::RAB_InformationSetup);
// RAB_Info_t rab_Info;
RBInfo *rb = masterConfig->getRB(rbid);
toAsnRAB_Info(rb,&result->rab_Info);
*pNumRBperRAB = rb->isCsDomain() ? 3 : 1;
// RB_InformationSetupList_t rb_InformationSetupList;
// A_SEQUENCE_OF(struct RB_InformationSetup) list
for (int i = 0; i < *pNumRBperRAB; i++, rbid++) {
rb = masterConfig->getRB(rbid);
// 3GPP 10.3.4.20
ASN::RB_InformationSetup *rbInfoIE = toAsnRB_InformationSetup(masterConfig, rb);
ASN_SEQUENCE_ADD(&result->rb_InformationSetupList.list,rbInfoIE);
}
return result;
}
// 3GPP 25.331 10.3.4.10 RAB Information for Setup.
static void toAsnRAB_InformationSetupList(RrcMasterChConfig *masterConfig, ASN::RAB_InformationSetupList *rbSetupListIE)
{
// For PS each RB is a RAB using one rb-id. There can be multiple RABs,
// although I think we will only setup one at a time because the UE can
// only ask for one at a time.
// For voice, the first 3 RBs constitute a RAB with 3 'sub-rab-flows' for the AMR codec.
// We dont allow mixing cs and ps connections on the same UE.
for (unsigned rbid = 5; rbid < masterConfig->mNumRB; ) {
RBInfo *rb = masterConfig->getRB(rbid);
if (!rb || ! rb->valid()) { continue; }
int numRBperRAB;
ASN::RAB_InformationSetup *rabSetupIE =
toAsnRAB_InformationSetup(masterConfig, rbid, &numRBperRAB);
ASN_SEQUENCE_ADD(&rbSetupListIE->list,rabSetupIE);
rbid += numRBperRAB; // Advance by the number of rbids used by this RAB.
}
}
static ASN::UL_CommonTransChInfo *toAsnUL_CommonTransChInfo(RrcMasterChConfig *masterConfig)
{
ASN::UL_CommonTransChInfo *result = RN_CALLOC(ASN::UL_CommonTransChInfo);
// struct TFC_Subset *tfc_Subset /* OPTIONAL */;
// struct TFCS *prach_TFCS /* OPTIONAL */;
// struct UL_CommonTransChInfo__modeSpecificInfo {...} modeSpecificInfo;
// The whole modeSpecificInfo is optional, even though it is not marked as such in ASN.
typedef ASN::UL_CommonTransChInfo::UL_CommonTransChInfo__modeSpecificInfo msi;
result->modeSpecificInfo = RN_CALLOC(msi);
result->modeSpecificInfo->present = ASN::UL_CommonTransChInfo__modeSpecificInfo_PR_fdd;
masterConfig->getUlTfcs()->toAsnTfcs(&result->modeSpecificInfo->choice.fdd.ul_TFCS,TrChUlDCHType);
return result;
}
static ASN::DL_CommonTransChInfo *toAsnDL_CommonTransChInfo(RrcMasterChConfig *masterConfig)
{
// struct DL_CommonTransChInfo
ASN::DL_CommonTransChInfo *result = RN_CALLOC(ASN::DL_CommonTransChInfo);
// struct TFCS *sccpch_TFCS /* OPTIONAL */;
// struct DL_CommonTransChInfo__modeSpecificInfo
// DL_CommonTransChInfo__modeSpecificInfo_PR present;
result->modeSpecificInfo.present = ASN::DL_CommonTransChInfo__modeSpecificInfo_PR_fdd;
// union DL_CommonTransChInfo__modeSpecificInfo_u {...} choice;
// struct DL_CommonTransChInfo__modeSpecificInfo__fdd {...} fdd;
// struct DL_CommonTransChInfo__modeSpecificInfo__fdd__dl_Parameters {...} *dl_Parameters;
typedef ASN::DL_CommonTransChInfo::
DL_CommonTransChInfo__modeSpecificInfo::
DL_CommonTransChInfo__modeSpecificInfo_u::
DL_CommonTransChInfo__modeSpecificInfo__fdd::
DL_CommonTransChInfo__modeSpecificInfo__fdd__dl_Parameters longthing;
result->modeSpecificInfo.choice.fdd.dl_Parameters = RN_CALLOC(longthing);
// This is where you can say same as uplink: (but we dont)
// DL_CommonTransChInfo__modeSpecificInfo__fdd__dl_Parameters_PR present;
result->modeSpecificInfo.choice.fdd.dl_Parameters->present =
ASN::DL_CommonTransChInfo__modeSpecificInfo__fdd__dl_Parameters_PR_dl_DCH_TFCS;
// union DL_CommonTransChInfo__modeSpecificInfo__fdd__dl_Parameters_u {...} choice;
// TFCS_t dl_DCH_TFCS;
masterConfig->getDlTfcs()->toAsnTfcs(
&result->modeSpecificInfo.choice.fdd.dl_Parameters->choice.dl_DCH_TFCS,TrChDlDCHType);
return result;
}
ASN::UL_AddReconfTransChInfoList *toAsnUL_AddReconfTransChInfoList(RrcMasterChConfig *masterConfig)
{
ASN::UL_AddReconfTransChInfoList *result = RN_CALLOC(ASN::UL_AddReconfTransChInfoList);
unsigned numtc = masterConfig->getUlNumTrCh();
// This assumes transport channels always start at 0.
for (TrChId tcid = 0; tcid < numtc; tcid++) {
//TrChInfo *tc = masterConfig->getUlTrChInfo(tcid);
// typedef struct UL_AddReconfTransChInformation
ASN::UL_AddReconfTransChInformation *addTcIE =
RN_CALLOC(ASN::UL_AddReconfTransChInformation);
// UL_TrCH_Type_t ul_TransportChannelType; // dch or usch.
asn_long2INTEGER(&addTcIE->ul_TransportChannelType,ASN::UL_TrCH_Type_dch);
// TransportChannelIdentity_t transportChannelIdentity;
addTcIE->transportChannelIdentity = tcid+1; // ASN TrCh starts at 1
// TransportFormatSet_t transportFormatSet;
masterConfig->getUlTfs()->toAsnTfs(&addTcIE->transportFormatSet);
ASN_SEQUENCE_ADD(&result->list,addTcIE);
}
return result;
}
// Create a fake uplink TrCh, needed just to align an ASN struct.
static void toAsnFakeUL_AddReconfTransChInfoList(ASN::UL_AddReconfTransChInfoList *ulchlist)
{
// The easiest way to do this is to manufacturer a TFS and use that:
UlTrChInfo ulfoo;
ulfoo.setTrCh(TrChRACHType,1,true);
ulfoo.getTfs()
->setSemiStatic(10,RrcDefs::Convolutional,ASN::CodingRate_half,256,16)
// ->setCommonCh() This did not work - failed to ASN encode.
->setDedicatedCh()
->addTF(0+4,0);
ASN::UL_AddReconfTransChInformation *dummyUlDCH = RN_CALLOC(ASN::UL_AddReconfTransChInformation);
asn_long2INTEGER(&dummyUlDCH->ul_TransportChannelType,ASN::UL_TrCH_Type_dch);
dummyUlDCH->transportChannelIdentity = 31;
// This does not work; asn_CHOICE does not except PR_NOTHING, anywhere!
//dummyUlDCH->transportFormatSet.present = ASN::TransportFormatSet_PR_NOTHING;
ulfoo.getTfs()->toAsnTfs(&dummyUlDCH->transportFormatSet);
ASN_SEQUENCE_ADD(&ulchlist->list,dummyUlDCH);
}
static ASN::DL_AddReconfTransChInfoList *toAsnDL_AddReconfTransChInfoList(RrcMasterChConfig *masterConfig)
{
ASN::DL_AddReconfTransChInfoList *result = RN_CALLOC(ASN::DL_AddReconfTransChInfoList);
unsigned numtc = masterConfig->getDlNumTrCh();
// This assumes transport channels always start at 0.
for (TrChId tcid = 0; tcid < numtc; tcid++) {
//TrChInfo *tc = getDlTrChInfo(tcid);
// v typedef struct DL_AddReconfTransChInformation
ASN::DL_AddReconfTransChInformation *addTcIE =
RN_CALLOC(ASN::DL_AddReconfTransChInformation);
// . DL_TrCH_Type_t dl_TransportChannelType;
asn_long2INTEGER(&addTcIE->dl_TransportChannelType,ASN::DL_TrCH_Type_dch);
// TransportChannelIdentity_t dl_transportChannelIdentity;
addTcIE->dl_transportChannelIdentity = tcid+1;
// vv struct DL_AddReconfTransChInformation__tfs_SignallingMode {...} tfs_SignallingMode;
// .. DL_AddReconfTransChInformation__tfs_SignallingMode_PR present;
addTcIE->tfs_SignallingMode.present = ASN::DL_AddReconfTransChInformation__tfs_SignallingMode_PR_explicit_config;
// vvv union DL_AddReconfTransChInformation__tfs_SignallingMode_u {...} choice;
// ... TransportFormatSet_t explicit_config;
// vvvv typedef struct TransportFormatSet
masterConfig->getDlTfs()->toAsnTfs(&addTcIE->tfs_SignallingMode.choice.explicit_config);
// ^^^^ struct TransportFormatSet
// ^^^ union DL_AddReconfTransChInformation__tfs_SignallingMode_u {...} choice;
// .. struct QualityTarget *dch_QualityTarget /* OPTIONAL */;
// .. struct TM_SignallingInfo *dummy /* OPTIONAL */;
// ^^ struct DL_AddReconfTransChInformation__tfs_SignallingMode {...} tfs_SignallingMode;
ASN_SEQUENCE_ADD(&result->list,addTcIE);
// ^ struct DL_AddReconfTransChInfoList
}
return result;
}
void toAsnDL_AddReconfTransChInfoListSameAsUl(
ASN::DL_AddReconfTransChInfoList *dlchlist,
int dltcid, // Downlink 1-based TrCh-id to configure
int ultcid) // Uplink 1-based TrCh-id to copy to downlink trch.
{
ASN::DL_AddReconfTransChInformation *dummyDlDCH =
RN_CALLOC(ASN::DL_AddReconfTransChInformation);
asn_long2INTEGER(&dummyDlDCH->dl_TransportChannelType,ASN::DL_TrCH_Type_dch);
dummyDlDCH->dl_transportChannelIdentity = dltcid;
// Since this is just a dummy structure, use the simplest type,which is sameAsUlTrCh.
dummyDlDCH->tfs_SignallingMode.present = ASN::DL_AddReconfTransChInformation__tfs_SignallingMode_PR_sameAsULTrCH;
asn_long2INTEGER(&dummyDlDCH->tfs_SignallingMode.choice.sameAsULTrCH.ul_TransportChannelType,
ASN::UL_TrCH_Type_dch);
dummyDlDCH->tfs_SignallingMode.choice.sameAsULTrCH.ul_TransportChannelIdentity = ultcid;
ASN_SEQUENCE_ADD(&dlchlist->list,dummyDlDCH);
}
ByteVector* sendDirectTransfer(UEInfo* uep, ByteVector &dlpdu, const char *descr, bool psDomain)
{
ASN::DL_DCCH_Message_t msg;
memset(&msg,0,sizeof(msg));
msg.message.present = ASN::DL_DCCH_MessageType_PR_downlinkDirectTransfer;
ASN::DownlinkDirectTransfer_t *ddt = &msg.message.choice.downlinkDirectTransfer;
ddt->present = ASN::DownlinkDirectTransfer_PR_r3;
ASN::DownlinkDirectTransfer_r3_IEs *ies = &ddt->choice.r3.downlinkDirectTransfer_r3;
ies->rrc_TransactionIdentifier = uep->newTransactionId();
ies->cn_DomainIdentity = psDomain ? toAsnEnumerated(ASN::CN_DomainIdentity_ps_domain) : toAsnEnumerated(ASN::CN_DomainIdentity_cs_domain);
ies->nas_Message.buf = dlpdu.begin();
ies->nas_Message.size = dlpdu.size();
ByteVector *result = new ByteVector(dlpdu.size()+100);
RN_MEMLOG(ByteVector,result);
if (!encodeDcchMsg(uep,SRB3,&msg,*result,descr)) {return NULL;}
return result;
}
static void toAsnURNTI(ASN::U_RNTI_t *urnti,unsigned srncid,unsigned srnti)
{
setAsnBIT_STRING(&urnti->srnc_Identity,(uint8_t*)calloc(1,2),12);
AsnBitString2BVTemp(urnti->srnc_Identity).setField(0,srncid,12);
setAsnBIT_STRING(&urnti->s_RNTI,(uint8_t*)calloc(1,3),20);
AsnBitString2BVTemp(urnti->s_RNTI).setField(0,srnti,20);
}
static ASN::C_RNTI_t *toAsnCRNTI(unsigned crnti)
{
// C_RNTI_t *new_c_RNTI /* OPTIONAL */;
ASN::C_RNTI_t *result = RN_CALLOC(ASN::C_RNTI_t);
// new_c_RNTI is a BIT_STRING_t
setAsnBIT_STRING(result,(uint8_t*)calloc(1,2),16);
AsnBitString2BVTemp(result).setField(0,crnti,16);
return result;
}
// NOTE: The RRC Connection Setup messages are defined as using CCCH and SRB0 which is TM
// uplink and UM downlink. That makes sense because uplink messages are small and the downlink
// message is huge and may need to be segmented.
// Note that for CCCH downlink messages the MAC does not include a UE-id, in fact,
// it doesn't have one yet because the RRC Connection Setup message is what is going
// to supply the URNTI and CRNTI; rather the UE-id is included in the message.
// Apparently the UE monitors the entire CCCH all the time in order to gather all the
// segments of all the messages and must look at them all to see if they are for it.
// Note that this message is still small: the rlc-info for TM&UM is pretty small,
// and the TrCh info comes from SIB5/6.
// Release 3 version of this message. The samsung and other phones did not seem to like this,
// so 11-16-2012 tried switching to release 4 version.
void sendRrcConnectionSetup(UEInfo *uep, ASN::InitialUE_Identity *ueInitialId)
{
ASN::DL_CCCH_Message msg;
memset(&msg,0,sizeof(msg));
msg.message.present = ASN::DL_CCCH_MessageType_PR_rrcConnectionSetup;
ASN::RRCConnectionSetup *csp = &msg.message.choice.rrcConnectionSetup;
unsigned transactionId = uep->newTransactionId();
ByteVector result(1000);
bool version4 = false; // Use version4 of this message.
bool version5 = false; // Use version5 of this message with a default config.
if (version4 | version5) {
csp->present = ASN::RRCConnectionSetup_PR_later_than_r3; // Guessing we can use any of the variants.
// WARNING: Now there are temporarily two pointers to the memory in UE_Identity.
csp->choice.later_than_r3.initialUE_Identity = *ueInitialId;
csp->choice.later_than_r3.rrc_TransactionIdentifier = transactionId;
//ASN::RRCConnectionSetup::RRCConnectionSetup_u::RRCConnectionSetup__later_than_r3 *iep =
//&csp->choice.later_than_r3;
//iep->initialUE_Identity = *ueInitialId;
//iep->rrc_TransactionIdentifier = transactionId;
//typedef ASN::RRCConnectionSetup::RRCConnectionSetup_u::RRCConnectionSetup__later_than_r3 later_than_r3_t;
//ASN::RRCConnectionSetup::RRCConnectionSetup_u::RRCConnectionSetup__later_than_r3::RRCConnectionSetup__later_than_r3__criticalExtensions *extp = &iep->criticalExtensions;
if (version5) {
// version 5 of the message. A little verbose here...
csp->choice.later_than_r3.criticalExtensions.present = ASN::RRCConnectionSetup__later_than_r3__criticalExtensions_PR_criticalExtensions;
csp->choice.later_than_r3.criticalExtensions.choice.criticalExtensions.present = ASN::RRCConnectionSetup__later_than_r3__criticalExtensions__criticalExtensions_PR_r5;
ASN::RRCConnectionSetup_r5_IEs_t *ie5 = &csp->choice.later_than_r3.criticalExtensions.choice.criticalExtensions.choice.r5.rrcConnectionSetup_r5;
// ActivationTime_t *activationTime /* OPTIONAL */;
// U_RNTI_t new_U_RNTI;
toAsnURNTI(&ie5->new_U_RNTI,uep->getSrncId(),uep->getSRNTI());
// C_RNTI_t *new_c_RNTI /* OPTIONAL */;
ie5->new_c_RNTI = toAsnCRNTI(uep->mCRNTI);
// RRC_StateIndicator_t rrc_StateIndicator;
asn_long2INTEGER(&ie5->rrc_StateIndicator,ASN::RRC_StateIndicator_cell_FACH);
// UTRAN_DRX_CycleLengthCoefficient_t utran_DRX_CycleLengthCoeff;
ie5->utran_DRX_CycleLengthCoeff = 3; // Must be in range 3..9
ie5->specificationMode.present = ASN::RRCConnectionSetup_r5_IEs__specificationMode_PR_preconfiguration;
ie5->specificationMode.choice.preconfiguration.preConfigMode.present =
ASN::RRCConnectionSetup_r5_IEs__specificationMode__preconfiguration__preConfigMode_PR_defaultConfig;
asn_long2INTEGER(&ie5->specificationMode.choice.preconfiguration.preConfigMode.choice.defaultConfig.defaultConfigMode,ASN::DefaultConfigMode_fdd);
ie5->specificationMode.choice.preconfiguration.preConfigMode.choice.defaultConfig.defaultConfigIdentity = 0;
} else {
// version 4 of the message.
csp->choice.later_than_r3.criticalExtensions.present = ASN::RRCConnectionSetup__later_than_r3__criticalExtensions_PR_r4;
ASN::RRCConnectionSetup_r4_IEs_t *ie4 = &csp->choice.later_than_r3.criticalExtensions.choice.r4.rrcConnectionSetup_r4;
// ActivationTime_t *activationTime /* OPTIONAL */;
// U_RNTI_t new_U_RNTI;
toAsnURNTI(&ie4->new_U_RNTI,uep->getSrncId(),uep->getSRNTI());
// C_RNTI_t *new_c_RNTI /* OPTIONAL */;
ie4->new_c_RNTI = toAsnCRNTI(uep->mCRNTI);
// RRC_StateIndicator_t rrc_StateIndicator;
// 11-17-2012: Tried cell_pch state instead, to see if Samsung likes that better.
// Nope. Still sends protocol error.
asn_long2INTEGER(&ie4->rrc_StateIndicator,ASN::RRC_StateIndicator_cell_FACH);
// UTRAN_DRX_CycleLengthCoefficient_t utran_DRX_CycleLengthCoeff;
ie4->utran_DRX_CycleLengthCoeff = 3; // Must be in range 3..9
// struct CapabilityUpdateRequirement_r4 *capabilityUpdateRequirement /* OPTIONAL */;
// SRB_InformationSetupList2_t srb_InformationSetupList;
toAsnSRB_InformationSetupList(gRrcDcchConfig, &ie4->srb_InformationSetupList.list);
if (0) {
// As a debug measure to try to get the Samsung to work, try putting in the TrCh items.
// But this is just not right, the DL_CommonTransChInfo IEs dont even have an option of setting up a FACH TrCh config.
// struct UL_CommonTransChInfo_r4 *ul_CommonTransChInfo /* OPTIONAL */;
ie4->ul_CommonTransChInfo = RN_CALLOC(ASN::UL_CommonTransChInfo_r4);
typedef ASN::UL_CommonTransChInfo_r4::UL_CommonTransChInfo_r4__modeSpecificInfo ulmode;
ie4->ul_CommonTransChInfo->modeSpecificInfo = RN_CALLOC(ulmode);
ie4->ul_CommonTransChInfo->modeSpecificInfo->present = ASN::UL_CommonTransChInfo_r4__modeSpecificInfo_PR_fdd;
gRrcDcchConfig->getUlTfcs()->toAsnTfcs(&ie4->ul_CommonTransChInfo->modeSpecificInfo->choice.fdd.ul_TFCS,TrChUlDCHType);
// struct UL_AddReconfTransChInfoList *ul_AddReconfTransChInfoList /* OPTIONAL */;
ie4->ul_AddReconfTransChInfoList = toAsnUL_AddReconfTransChInfoList(gRrcDcchConfig);
// struct DL_CommonTransChInfo_r4 *dl_CommonTransChInfo /* OPTIONAL */;
ie4->dl_CommonTransChInfo = RN_CALLOC(ASN::DL_CommonTransChInfo_r4);
typedef ASN::DL_CommonTransChInfo_r4::DL_CommonTransChInfo_r4__modeSpecificInfo dlmode;
ie4->dl_CommonTransChInfo->modeSpecificInfo = RN_CALLOC(dlmode);
ie4->dl_CommonTransChInfo->modeSpecificInfo->present = ASN::DL_CommonTransChInfo_r4__modeSpecificInfo_PR_fdd;
typedef ASN::DL_CommonTransChInfo_r4::
DL_CommonTransChInfo_r4__modeSpecificInfo::
DL_CommonTransChInfo_r4__modeSpecificInfo_u::
DL_CommonTransChInfo_r4__modeSpecificInfo__fdd::
DL_CommonTransChInfo_r4__modeSpecificInfo__fdd__dl_Parameters dl_ridiculous;
ie4->dl_CommonTransChInfo->modeSpecificInfo->choice.fdd.dl_Parameters = RN_CALLOC(dl_ridiculous);
ie4->dl_CommonTransChInfo->modeSpecificInfo->choice.fdd.dl_Parameters->present =
ASN::DL_CommonTransChInfo_r4__modeSpecificInfo__fdd__dl_Parameters_PR_sameAsUL;
// struct DL_AddReconfTransChInfoList_r4 *dl_AddReconfTransChInfoList /* OPTIONAL */;
//gRrcCcchConfig->getDlTfs()->toAsnTfs(&fachPchInfo->transportFormatSet);
//gRrcCcchConfig->getUlTfs()->toAsnTfs((prach_SI->rach_TransportFormatSet = RN_CALLOC(ASN::TransportFormatSet)));
}
// struct FrequencyInfo *frequencyInfo /* OPTIONAL */;
// MaxAllowedUL_TX_Power_t *maxAllowedUL_TX_Power /* OPTIONAL */;
// struct UL_ChannelRequirement_r4 *ul_ChannelRequirement /* OPTIONAL */;
// struct DL_CommonInformation_r4 *dl_CommonInformation /* OPTIONAL */;
// struct DL_InformationPerRL_List_r4 *dl_InformationPerRL_List /* OPTIONAL */;
} // version 4
if (!encodeCcchMsg(&msg,result,descrRrcConnectionSetup,uep,0)) {return;}
// Zero out the initialUE_Identity that we copied above so that there
// is only one copy of it and we dont try to free it twice.
memset(&csp->choice.later_than_r3.initialUE_Identity,0,sizeof(ASN::InitialUE_Identity_t));
} else {
// Version 3 of this message.
// struct RRCConnectionSetup_r3_IEs
csp->present = ASN::RRCConnectionSetup_PR_r3; // Guessing we can use any of the variants.
ASN::RRCConnectionSetup_r3_IEs_t *iep = &csp->choice.r3.rrcConnectionSetup_r3;
// InitialUE_Identity_t initialUE_Identity;
// WARNING: Now there are temporarily two pointers to the memory in UE_Identity.
iep->initialUE_Identity = *ueInitialId;
// RRC_TransactionIdentifier_t rrc_TransactionIdentifier;
iep->rrc_TransactionIdentifier = transactionId;
// ActivationTime_t *activationTime /* OPTIONAL */;
// U_RNTI_t new_U_RNTI;
// U-RNTI is mandatory.
// They took apart the U_RNTI into its constituent parts, which was kinda dumb:
// the parts are 12 bit SRNC id and 20 bit S-RNTI.
toAsnURNTI(&iep->new_U_RNTI,uep->getSrncId(),uep->getSRNTI());
//setAsnBIT_STRING(&iep->new_U_RNTI.srnc_Identity,(uint8_t*)calloc(1,2),12);
//AsnBitString2BVTemp(iep->new_U_RNTI.srnc_Identity).setField(0,uep->getSrncId(),12);
//setAsnBIT_STRING(&iep->new_U_RNTI.s_RNTI,(uint8_t*)calloc(1,3),20);
//AsnBitString2BVTemp(iep->new_U_RNTI.s_RNTI).setField(0,uep->getSRNTI(),20);
// srnti = 0x12345;
//ByteVector tst(3);
//tst.setField(0,srnti,20);
//printf("SRNTI=0x%x bv=%s\n",srnti,tst.hexstr().c_str());
// C_RNTI_t *new_c_RNTI /* OPTIONAL */;
// C-RNTI
iep->new_c_RNTI = toAsnCRNTI(uep->mCRNTI);
//iep->new_c_RNTI = RN_CALLOC(ASN::C_RNTI_t);
// new_c_RNTI is a BIT_STRING_t
//setAsnBIT_STRING(iep->new_c_RNTI,(uint8_t*)calloc(1,2),16);
//AsnBitString2BVTemp(iep->new_c_RNTI).setField(0,uep->mCRNTI,16);
// RRC_StateIndicator_t rrc_StateIndicator;
asn_long2INTEGER(&iep->rrc_StateIndicator,ASN::RRC_StateIndicator_cell_FACH);
// UTRAN_DRX_CycleLengthCoefficient_t utran_DRX_CycleLengthCoeff;
// 10.3.3.49: DRC mode. "Refers to 'k' in the formula 25.304 Discontinous Reception".
iep->utran_DRX_CycleLengthCoeff = 3; // Must be in range 3..9
// skip optional CapabilityUpdateRequirement
// struct CapabilityUpdateRequirement *capabilityUpdateRequirement /* OPTIONAL */;
iep->capabilityUpdateRequirement = RN_CALLOC(ASN::CapabilityUpdateRequirement);
iep->capabilityUpdateRequirement->ue_RadioCapabilityFDDUpdateRequirement = true;
iep->capabilityUpdateRequirement->ue_RadioCapabilityTDDUpdateRequirement = false;
// SRB_InformationSetupList2_t srb_InformationSetupList;
toAsnSRB_InformationSetupList(gRrcDcchConfig, &iep->srb_InformationSetupList.list);
// struct UL_CommonTransChInfo *ul_CommonTransChInfo /* OPTIONAL */;
// struct DL_CommonInformation *dl_CommonInformation /* OPTIONAL */;
// UL_AddReconfTransChInfoList_t ul_AddReconfTransChInfoList;
// DL_AddReconfTransChInfoList_t dl_AddReconfTransChInfoList;
// All the TrCh info is optional, and not used in our case because we are
// defining RACH/FACH rather than DCH, BUT...
// For ul_ and dl_AddReconfTransChInfoList we are required to put in something anyway,
// and 8.1.3.4 recommends a single zero-sized TF.
// You can not use the "NOTHING" option - the ASN compiler just uses
// that to mark an uninitialized value and fails.
toAsnFakeUL_AddReconfTransChInfoList(&iep->ul_AddReconfTransChInfoList);
toAsnDL_AddReconfTransChInfoListSameAsUl(&iep->dl_AddReconfTransChInfoList,31,31);
// These IEs are all skipped, needed only for DCH:
// struct UL_ChannelRequirement *ul_ChannelRequirement /* OPTIONAL */;
// struct DL_InformationPerRL_List *dl_InformationPerRL_List /* OPTIONAL */;
//PhCh *phch = new PhCh(DPDCHType,256,254,256,9999,NULL);
//iep->ul_ChannelRequirement = phch->toAsnUL_ChannelRequirement();
//iep->dl_CommonInformation = phch->toAsnDL_CommonInformation();
//iep->dl_InformationPerRL_List = phch->toAsnDL_InformationPerRL_List();
/*ASN::DL_InformationPerRL_List *result2 = RN_CALLOC(ASN::DL_InformationPerRL_List);
ASN::DL_InformationPerRL *one = RN_CALLOC(ASN::DL_InformationPerRL);
one->modeSpecificInfo.present = ASN::DL_InformationPerRL__modeSpecificInfo_PR_fdd;
int primarySC = gConfig.getNum("UMTS.Downlink.ScramblingCode");
one->modeSpecificInfo.choice.fdd.primaryCPICH_Info.primaryScramblingCode = primarySC;
//one->dl_DPCH_InfoPerRL = toAsnDL_DPCH_InfoPerRL();
ASN_SEQUENCE_ADD(&result2->list,one);
iep->dl_InformationPerRL_List = result2;*/
// struct FrequencyInfo *frequencyInfo /* OPTIONAL */;
// TODO: Do we need this?
// MaxAllowedUL_TX_Power_t *maxAllowedUL_TX_Power /* OPTIONAL */;
if (!encodeCcchMsg(&msg,result,descrRrcConnectionSetup,uep,0)) {return;}
// Zero out the initialUE_Identity that we copied above so that there
// is only one copy of it and we dont try to free it twice.
memset(&iep->initialUE_Identity,0,sizeof(ASN::InitialUE_Identity_t));
}
LOG(INFO) << "gNodeB: " << gNodeB.clock().get() << ", SCCPCH: " << result;
// TODO: This crashes
// ASN_STRUCT_FREE_CONTENTS_ONLY(ASN::asn_DEF_DL_CCCH_Message,&msg);
// Configure the UE to be ready for incoming on the new SRBs...
uep->ueConnectRlc(gRrcDcchConfig,stCELL_FACH);
// Set a UeTransaction: see ueWriteLowSide for its use.
// there is no chance that other unrelated messages can happen simultaneously.
// But we will print an error if the received transaction does not match.
// The Rab Mask parameter is not relevant for this transaction type.
UeTransaction(uep,UeTransaction::ttRrcConnectionSetup, 0, transactionId,stCELL_FACH);
gMacSwitch.writeHighSideCcch(result,descrRrcConnectionSetup);
}
// Sent when an unrecognized UE tries to talk to us.
// Tell it to release the connection and start over.
static void sendRrcConnectionReleaseCcch(int32_t urnti)
{
ASN::DL_CCCH_Message_t msg;
memset(&msg,0,sizeof(msg));
msg.message.present = ASN::DL_CCCH_MessageType_PR_rrcConnectionRelease;
ASN::RRCConnectionRelease_CCCH *m1 = &msg.message.choice.rrcConnectionRelease;
m1->present = ASN::RRCConnectionRelease_CCCH_PR_r3;
ASN::RRCConnectionRelease_CCCH_r3_IEs *m2 = &m1->choice.r3.rrcConnectionRelease_CCCH_r3;
unsigned srncid = urnti >> 20 & 0xfff;
unsigned srnti = urnti & 0xfffff;
toAsnURNTI(&m2->u_RNTI, srncid,srnti);
ASN::RRCConnectionRelease_r3_IEs_t *m3 = &m2->rrcConnectionRelease;
m3->rrc_TransactionIdentifier = 0; // bogus, because we are not expecting a reply.
m3->releaseCause = toAsnEnumerated(ASN::ReleaseCause_unspecified); // TODO: What cause should we use?
ByteVector result(1000);
if (!encodeCcchMsg(&msg,result,descrRrcConnectionRelease,NULL,urnti)) {return;}
gMacSwitch.writeHighSideCcch(result,descrRrcConnectionRelease);
}
// This puts the phone in idle mode.
void sendRrcConnectionRelease(UEInfo *uep) //, ASN::InitialUE_Identity *ueInitialId
{
// Create the RB Setup Message.
ASN::DL_DCCH_Message_t msg;
memset(&msg,0,sizeof(msg));
msg.message.present = ASN::DL_DCCH_MessageType_PR_rrcConnectionRelease;
msg.message.choice.rrcConnectionRelease.present = ASN::RRCConnectionRelease_PR_r3;
ASN::RRCConnectionRelease_r3_IEs_t *ies =
&msg.message.choice.rrcConnectionRelease.choice.r3.rrcConnectionRelease_r3;
// Note: The RRC spec has a U-RNTI in the message, but it is not in the ASN.
// I do not see why it would be needed. If the message goes on DCCH the
// U-RNTI will be in the MAC header.
// RRC_TransactionIdentifier_t rrc_TransactionIdentifier;
unsigned transactionId = uep->newTransactionId();
ies->rrc_TransactionIdentifier = transactionId;
// N_308_t *n_308 /* OPTIONAL */;
// N308 is the number of times UE sends response RrcConnectionReleaseComplete.
// It is mandatory when this message is sent on DCCH.
ies->n_308 = RN_CALLOC(ASN::N_308_t); // It is a long.
*ies->n_308 = 1; // must be 1..8
// ReleaseCause_t releaseCause;
// Causes we might use are: normalEvent, userInactivity, pre_emptiveRelease.
ies->releaseCause = toAsnEnumerated(ASN::ReleaseCause_normalEvent);
// struct Rplmn_Information *rplmn_information /* OPTIONAL */;
ByteVector result(1000);
if (!encodeDcchMsg(uep,SRB2,&msg,result,descrRrcConnectionRelease)) {return;}
// Prepare to receive the reply to this message:
UeTransaction(uep,UeTransaction::ttRrcConnectionRelease,0,transactionId,stIdleMode);
uep->ueWriteHighSide(SRB2, result, descrRrcConnectionRelease);
}
// 3GPP 25.331 10.2.33
// This is the main message to create DCH channels.
// It is invoked by the SGSN to create a RAB for an internet connection.
// It will also be invoked by GMM L3 to create CS connections.
// In both cases a state machine in the caller must wait for the phones
// response before proceeding.
//
// We are sending a setup for a DCH and moving the UE to CELL_DCH state.
// The masterConfig indicates the DCH L2 setup, for example, how many TrCh.
// Return non-zero on error.
bool sendRadioBearerSetup(UEInfo *uep, RrcMasterChConfig *masterConfig, PhCh *phch, bool srbstoo)
{
// Create the RB Setup Message.
ASN::DL_DCCH_Message_t msg;
memset(&msg,0,sizeof(msg));
msg.message.present = ASN::DL_DCCH_MessageType_PR_radioBearerSetup;
ASN::RadioBearerSetup_t &rbstop = msg.message.choice.radioBearerSetup;
// There are various versions of this message. Lets use the oldest version:
rbstop.present = ASN::RadioBearerSetup_PR_r3;
ASN::RadioBearerSetup_r3_IEs_t &rbs = rbstop.choice.r3.radioBearerSetup_r3;
// =============== UE Information Elements ==============
// Comments are directly from the ASN::RadioBearerSetup_r3_IEs_t
// RRC_TransactionIdentifier_t rrc_TransactionIdentifier;
unsigned transactionId = uep->newTransactionId();
rbs.rrc_TransactionIdentifier = transactionId;
// struct IntegrityProtectionModeInfo *integrityProtectionModeInfo /* OPTIONAL */;
// struct CipheringModeInfo *cipheringModeInfo /* OPTIONAL */;
// ActivationTime_t *activationTime /* OPTIONAL */;
// struct U_RNTI *new_U_RNTI /* OPTIONAL */;
// C_RNTI_t *new_C_RNTI /* OPTIONAL */;
// RRC_StateIndicator_t rrc_StateIndicator;
asn_long2INTEGER(&rbs.rrc_StateIndicator, ASN::RRC_StateIndicator_cell_DCH);
// UTRAN_DRX_CycleLengthCoefficient_t *utran_DRX_CycleLengthCoeff /* OPTIONAL */;
// URA_Identity_t *ura_Identity /* OPTIONAL */;
// struct CN_InformationInfo *cn_InformationInfo /* OPTIONAL */;
// =============== RB Information Elements ==============
// Set up new SRBs on the new DCH.
// The new SRBs will be multiplexed using the paradigm of the defaultRBMappingInfo
// struct SRB_InformationSetupList *srb_InformationSetupList /* OPTIONAL */;
if (srbstoo) {
// 3GPP 25.331 10.3.4.24
rbs.srb_InformationSetupList = RN_CALLOC(ASN::SRB_InformationSetupList);
toAsnSRB_InformationSetupList(masterConfig,&rbs.srb_InformationSetupList->list);
}
// The RAB is where the new RBID for the data channel needs to go.
// struct RAB_InformationSetupList *rab_InformationSetupList /* OPTIONAL */;
// The UE is allowed to request a PS channel on any of RBs 5..15, so check them all.
bool haveDataCh = false; // Does the config define any data channels?
RBInfo *rb; unsigned rbid;
for (rbid = 5; rbid < masterConfig->mNumRB; rbid++) {
if (!(rb = masterConfig->getRB(rbid)) || !rb->valid()) { continue; }
haveDataCh = true;
break;
}
if (haveDataCh) {
// 3GPP 25.331 10.3.4.10 RAB Information for Setup.
// TODO: AMR rate defaults to "t7" - what is that?
rbs.rab_InformationSetupList = RN_CALLOC(ASN::RAB_InformationSetupList);
// A_SEQUENCE_OF(struct RAB_InformationSetup) list;
toAsnRAB_InformationSetupList(masterConfig, rbs.rab_InformationSetupList);
}
// struct RB_InformationAffectedList *rb_InformationAffectedList /* OPTIONAL */;
// struct DL_CounterSynchronisationInfo *dl_CounterSynchronisationInfo /* OPTIONAL */;
// =============== Uplink Transport Channels ==============
// struct UL_CommonTransChInfo *ul_CommonTransChInfo /* OPTIONAL */;
rbs.ul_CommonTransChInfo = toAsnUL_CommonTransChInfo(masterConfig);
// struct UL_DeletedTransChInfoList *ul_deletedTransChInfoList /* OPTIONAL */;
// struct UL_AddReconfTransChInfoList *ul_AddReconfTransChInfoList /* OPTIONAL */;
rbs.ul_AddReconfTransChInfoList = toAsnUL_AddReconfTransChInfoList(masterConfig);
// =============== Downlink Transport Channels ==============
// struct RadioBearerSetup_r3_IEs__dummy { } *dummy;
// struct DL_CommonTransChInfo *dl_CommonTransChInfo /* OPTIONAL */;
rbs.dl_CommonTransChInfo = toAsnDL_CommonTransChInfo(masterConfig);
// struct DL_DeletedTransChInfoList *dl_DeletedTransChInfoList /* OPTIONAL */;
// struct DL_AddReconfTransChInfoList *dl_AddReconfTransChInfoList /* OPTIONAL */;
rbs.dl_AddReconfTransChInfoList = toAsnDL_AddReconfTransChInfoList(masterConfig);
// =============== PhyCH Information Elements ==============
// struct FrequencyInfo *frequencyInfo /* OPTIONAL */;
// TODO: Do we need to add frequency info? For FDD is it is just the ARFCN,
// and I assume we are single ARFCN for now.
// =============== Uplink Radio Resources ==============
// MaxAllowedUL_TX_Power_t *maxAllowedUL_TX_Power /* OPTIONAL */;
// TODO: Do we need to specify UL power?
// This is a container for 10.3.6.88 "Uplink DPCH Info" in RRC spec.
// struct UL_ChannelRequirement *ul_ChannelRequirement /* OPTIONAL */;
// This is for DPCH only.
rbs.ul_ChannelRequirement = phch->toAsnUL_ChannelRequirement();
// struct RadioBearerSetup_r3_IEs__modeSpecificPhysChInfo { ... } modeSpecificPhysChInfo;
rbs.modeSpecificPhysChInfo.present = ASN::RadioBearerSetup_r3_IEs__modeSpecificPhysChInfo_PR_fdd;
// But the contents are optional.
// =============== Downlink Radio Resources ==============
// struct DL_CommonInformation *dl_CommonInformation /* OPTIONAL */;
// And I quote 8.2.1:
// If after state transition the UE enters CELL_DCH state from CELL_FACH or from CELL_PCH state:
// 1> if the IE "Default DPCH Offset Value" is not included:
// 2> the UE behaviour is not specified.
// The DPCH offset is inside this element:
// (with the totally unpredictable value of "0" which apparently the UE could never have guessed on its own.)
rbs.dl_CommonInformation = phch->toAsnDL_CommonInformation();
// struct DL_InformationPerRL_List *dl_InformationPerRL_List /* OPTIONAL */;
rbs.dl_InformationPerRL_List = phch->toAsnDL_InformationPerRL_List();
// note: encodeDcchMsg dumps the message to the log file.
//asn_fprint(stdout,&ASN::asn_DEF_DL_DCCH_Message, &msg); // Dump it all.
//fflush(stdout);
ByteVector result(1000);
if (!encodeDcchMsg(uep,SRB2,&msg,result,descrRadioBearerSetup)) {return 1;}
LOG(INFO) << "gNodeB: " << gNodeB.clock().get() << ", RadioBearerSetup: " << result;
// Prepare to receive the reply to this message:
UeTransaction(uep,UeTransaction::ttRadioBearerSetup, 1<<rbid, transactionId,stCELL_DCH);
// This message is being written into a queue, but the MAC could find
// it there and send it immediately. The caller is responsible
// for making sure the UE is prepared to receive replies before calling us.
uep->ueWriteHighSide(SRB2, result, descrRadioBearerSetup);
return 0; // ok
}
// 3GPPP 25.331 8.2.2
// pat 12-27-2012: This message is implemented incorrectly. I was hoping if you told the UE
// to go to CELL_FACH it would use the info from SIB5 or SIB6, but nope, you have to download
// a completely new config that is basically going to be a copy of SIB5.
// I'm not sure we have to do that at all, because when the last radio bearer is released
// we can send an RrcConnectionRelease to drop the UE back into idle mode.
// The UE keeps its SGSN level attach status regardless of mode; idle mode is fine.
// Update: Some phones (Samsung) activate/deactivate the PDPContexts rapidly, and we dont want
// to drop them all the way back to idle mode, because the transition from idle->fach takes a long
// time because we have to go through the authorization and security steps, while fach->dch is just a single message.
// Long term, we dont want to leave the UE in CELL_FACH mode after the radio-bearer-release anyway,
// it should be in CELL_PCH so we can page it, or if we dont need to page it, in idle mode.
// Harvind (3-17-13) I think this message is fine. The UE seems to like it during the PDP Deactivation Process
void sendRadioBearerRelease(UEInfo *uep,
unsigned rabMask, // Mask of RABs/RBs to release.
bool finished) // If set, all RABs released so move UE back to CELL_FACH mode.
{
#if 1
if (finished) {
sendRrcConnectionRelease(uep);
return;
}
//return; // Do nothing. We will just leave the existing DCH setup alone; it does not matter
#endif
// how many RBs are sharing the DCH.
UEState nextState = finished ? stCELL_FACH : stCELL_DCH;
ASN::DL_DCCH_Message_t msg;
memset(&msg,0,sizeof(msg));
msg.message.present = ASN::DL_DCCH_MessageType_PR_radioBearerRelease;
ASN::RadioBearerRelease_t &rbrtop = msg.message.choice.radioBearerRelease;
// There are various versions of this message. Lets use the oldest version:
rbrtop.present = ASN::RadioBearerRelease_PR_r3;
ASN::RadioBearerRelease_r3_IEs_t &rbr = rbrtop.choice.r3.radioBearerRelease_r3;
// =============== UE Information Elements ==============
//RRC_TransactionIdentifier_t rrc_TransactionIdentifier;
unsigned transactionId = uep->newTransactionId();
rbr.rrc_TransactionIdentifier = transactionId;
//struct IntegrityProtectionModeInfo *integrityProtectionModeInfo /* OPTIONAL */;
//struct CipheringModeInfo *cipheringModeInfo /* OPTIONAL */;
//ActivationTime_t *activationTime /* OPTIONAL */;
//struct U_RNTI *new_U_RNTI /* OPTIONAL */;
//C_RNTI_t *new_C_RNTI /* OPTIONAL */;
//RRC_StateIndicator_t rrc_StateIndicator;
rbr.rrc_StateIndicator = toAsnEnumerated(UEState2Asn(nextState));
//UTRAN_DRX_CycleLengthCoefficient_t *utran_DRX_CycleLengthCoeff /* OPTIONAL */;
// =============== CN Information Elements ==============
//struct CN_InformationInfo *cn_InformationInfo /* OPTIONAL */;
//CN_DomainIdentity_t *signallingConnectionRelIndication /* OPTIONAL */;
// =============== UTRAN Mobility Information Elements ==============
//URA_Identity_t *ura_Identity /* OPTIONAL */;
// =============== RB Information Elements ==============
// 8.6.4.5 says RB Information to Reconfigure can modify SRBs.
#if 0 // If you include this code you get an ASN encoder violation from encodeDcchMsg
// Furthermore, the examples I have seen dont use this, they release the RBs,
// and the UE must infer the RAB from the RB.
//struct RAB_InformationReconfigList *rab_InformationReconfigList /* OPTIONAL */;
for (RbId rbid = 5; rbid < gsMaxRB; rbid++) {
if (rabMask & (1<<rbid)) {
// 25.331 10.3.4.11 RAB Information to Reconfigure.
RBInfo *rb;
if (!(rb = uep->ueGetConfig()->getRB(rbid)) || !rb->valid()) { continue; }
ASN::RAB_InformationReconfig *rabreconfig = RN_CALLOC(ASN::RAB_InformationReconfig);
toAsnRAB_Identity(rbid,&rabreconfig->rab_Identity);
bool csdomain = rb->isCsDomain();
rabreconfig->cn_DomainIdentity = toAsnEnumerated(
csdomain ? ASN::CN_DomainIdentity_cs_domain : ASN::CN_DomainIdentity_ps_domain);
// NAS_Synchronisation_Indicator_t nas_Synchronisation_Indicator;
// From the documentation:
// A container for non-access stratum information to be transferred transparently through UTRAN.
// Note 1: The nas_Synchronisation_Indicator is only relevant for CS domains.
// On the first valid RAB, allocate the list.
if (rbr.rab_InformationReconfigList == NULL) {
rbr.rab_InformationReconfigList = RN_CALLOC(ASN::RAB_InformationReconfigList);
}
ASN_SEQUENCE_ADD(&rbr.rab_InformationReconfigList->list,rabreconfig);
}
}
#endif
//RB_InformationReleaseList_t rb_InformationReleaseList;
for (RbId rbid = 5; rbid < gsMaxRB; rbid++) {
if (rabMask & (1<<rbid)) {
ASN::RB_Identity_t *val = RN_CALLOC(ASN::RB_Identity_t); // Its just a long.
*val = rbid;
ASN_SEQUENCE_ADD(&rbr.rb_InformationReleaseList.list,val);
}
}
// The RB Information to Reconfigure IE is only in ASN version 6 and higher.
// The difference between that and RB Information to be Affected is that
// the former lets you change other RLC programming, while
// RB Information to be Affected lets you change only the RB Mapping Info,
// but that is all you need to switch the SRBs between CELL_DCH or CELL_FACH state.
//struct RB_InformationAffectedList *rb_InformationAffectedList /* OPTIONAL */;
// 12-22-2012: This message is not working, so tried removing this, but did not help:
if (finished) {
// Reconfigure the SRBs to whatever they need to be in CELL_FACH state.
rbr.rb_InformationAffectedList = toAsnRB_InformationAffectedList(gRrcDcchConfig);
}
//struct DL_CounterSynchronisationInfo *dl_CounterSynchronisationInfo /* OPTIONAL */;
// =============== TrCh Information Elements ==============
//struct UL_CommonTransChInfo *ul_CommonTransChInfo /* OPTIONAL */;
//struct UL_DeletedTransChInfoList *ul_deletedTransChInfoList /* OPTIONAL */;
//struct UL_AddReconfTransChInfoList *ul_AddReconfTransChInfoList /* OPTIONAL */;
//struct RadioBearerRelease_r3_IEs__dummy { } *dummy;
//struct DL_CommonTransChInfo *dl_CommonTransChInfo /* OPTIONAL */;
//struct DL_DeletedTransChInfoList *dl_DeletedTransChInfoList /* OPTIONAL */;
//struct DL_AddReconfTransChInfo2List *dl_AddReconfTransChInfoList /* OPTIONAL */;
// pat 1-24-2013: Now making the assumption that we use different TrCh for CELL_FACH and CELL_DCH mode,
// so when we are 'finished' with DCH we can just delete the TrCh used for CELL_DCH.
// In this case we still need to send an updated TFCS.
// If we are not 'finished' with the DCH, we dont touch it, because we use logical channel multiplexing
// to distinguish the various RBs on a single TrCh.
if (finished) {
// pat 1-24-2013:
// Apparently we are supposed to delete the TrCh being used by DCH. I still dont know if UE stores
// the two separate TFS for use in CELL_FACH and CELL_DCH modes; this critical piece of info is not in 25.331.
// To be safe, I would like to use different TrChs numbers for RACH/FACH and DCH, but
// unfortunately there are several places in our code that assume that all TrChId in use
// are numbered starting at 1 (TrCh ids are 1-based), so we cannot yet allocate a
// different TrCh for DCH than RACH/FACH, and I was hesitant to change all those places
// because I cannot test because no radio.
// So instead, just delete TrCh 1 and hope the UE reloads TrCh 1 for FACH from SIB5.
// EXCEPT: The SIB5 can (and currently does) have the wrong FACH TrCh numbers in it, and the UE apparently does not even care!
// The only thing I can imagine is that the UE is pretty much ignoring the TrCh ids in FACH state, implying that
// the spec is totally screwed up here and nobody knows how it is really supposed to work.
{
rbr.ul_deletedTransChInfoList = RN_CALLOC(ASN::UL_DeletedTransChInfoList_t);
ASN::UL_TransportChannelIdentity_t *ult = RN_CALLOC(ASN::UL_TransportChannelIdentity_t);
ult->ul_TransportChannelType = toAsnEnumerated(ASN::UL_TrCH_Type_dch);
// TODO: We are simply assuming that there is only one TrCh here.
//ult->ul_TransportChannelIdentity = uep->mUeDchConfig.getUlTrChInfo(0)->mTransportChannelIdentity;
ult->ul_TransportChannelIdentity = 1;
ASN_SEQUENCE_ADD(&rbr.ul_deletedTransChInfoList->list,ult);
}
{
rbr.dl_DeletedTransChInfoList = RN_CALLOC(ASN::DL_DeletedTransChInfoList_t);
ASN::DL_TransportChannelIdentity_t *dlt = RN_CALLOC(ASN::DL_TransportChannelIdentity_t);
dlt->dl_TransportChannelType = toAsnEnumerated(ASN::DL_TrCH_Type_dch);
// TODO: We are simply assuming that there is only the TrCh here.
//dlt->dl_TransportChannelIdentity = uep->mUeDchConfig.getDlTrChInfo(0)->mTransportChannelIdentity;
dlt->dl_TransportChannelIdentity = 1;
ASN_SEQUENCE_ADD(&rbr.dl_DeletedTransChInfoList->list,dlt);
}
// The only example RadioBearerRelease message example I have is Nokia's "Call Setup PS" but that
// leaves the UE in CELL_DCH mode afterward, not CELL_FACH. They download a new TFCS but not a TFS.
// I dont think we should do this unless we are leaving the UE in CELL_DCH state too.
RrcMasterChConfig *newConfig = gRrcDcchConfig;
rbr.ul_CommonTransChInfo = toAsnUL_CommonTransChInfo(newConfig);
// The Add/Reconfig TrChInfoList is only for DCHs, not FACH/RACHes
// Therefore it must pick up the FACH TFS from SIB5, and It is totally nonsensical to me why
// we must download a TFCS here instead of the UE using the one from SIB5.
//rbr.ul_AddReconfTransChInfoList = toAsnUL_AddReconfTransChInfoList(newConfig);
rbr.dl_CommonTransChInfo = toAsnDL_CommonTransChInfo(newConfig);
//rbr.dl_AddReconfTransChInfo2List = toAsnDL_AddReconfTransChInfo2List(newConfig);
}
// =============== Physical Information Elements ============== We dont need these.
//struct FrequencyInfo *frequencyInfo /* OPTIONAL */;
//MaxAllowedUL_TX_Power_t *maxAllowedUL_TX_Power /* OPTIONAL */;
//struct UL_ChannelRequirement *ul_ChannelRequirement /* OPTIONAL */;
//struct RadioBearerRelease_r3_IEs__modeSpecificPhysChInfo { } modeSpecificPhysChInfo;
// Must set present, but the contents are optional.
rbr.modeSpecificPhysChInfo.present = ASN::RadioBearerRelease_r3_IEs__modeSpecificPhysChInfo_PR_fdd;
//struct DL_CommonInformation *dl_CommonInformation /* OPTIONAL */;
//struct DL_InformationPerRL_List *dl_InformationPerRL_List /* OPTIONAL */;
ByteVector result(1000);
if (!encodeDcchMsg(uep,SRB2,&msg,result,descrRadioBearerRelease)) {return;}
if (finished) {
// Configure the UE to be ready for incoming on the new SRBs...
uep->ueConnectRlc(gRrcDcchConfig,nextState);
}
// Prepare to receive the reply to this message:
UeTransaction(uep,UeTransaction::ttRadioBearerRelease, rabMask, transactionId,
nextState);
uep->ueWriteHighSide(SRB2, result, descrRadioBearerRelease);
}
static unsigned commonCellUpdateConfirm(UEInfo *uep, ASN::CellUpdateConfirm_r3_IEs_t *ies)
{
// Apparently even CCCH messages get a transaction id, which will be used if the UE
// replies to indicate an error in this message.
unsigned transactionId = uep->newTransactionId();
ies->rrc_TransactionIdentifier = transactionId;
// Huge message but almost everything is optional. Here are the mandatory parts:
ies->rrc_StateIndicator = toAsnEnumerated(UEState2Asn(uep->ueGetState()));
ies->rlc_Re_establishIndicatorRb2_3or4 = 0;
ies->rlc_Re_establishIndicatorRb5orAbove = 0;
ies->modeSpecificTransChInfo.present = ASN::CellUpdateConfirm_r3_IEs__modeSpecificTransChInfo_PR_fdd;
ies->modeSpecificPhysChInfo.present = ASN::CellUpdateConfirm_r3_IEs__modeSpecificPhysChInfo_PR_fdd;
// We can define a new URNTI.
// TODO: Do we need to assign a new URNTI if this message is on CCCH?
return transactionId;
}
// The CellUpdateConfirm message may be sent out on either DCCH or CCCH.
// This version is for DCCH.
// TODO: It would be wise to implement the RLC re-establish indicators.
static void sendCellUpdateConfirmDcch(UEInfo *uep)
{
ASN::DL_DCCH_Message_t msg;
memset(&msg,0,sizeof(msg));
// struct IntegrityCheckInfo *integrityCheckInfo /* OPTIONAL */;
msg.message.present = ASN::DL_DCCH_MessageType_PR_cellUpdateConfirm;
msg.message.choice.cellUpdateConfirm.present = ASN::CellUpdateConfirm_PR_r3;
ASN::CellUpdateConfirm_r3_IEs_t *ies =
&msg.message.choice.cellUpdateConfirm.choice.r3.cellUpdateConfirm_r3;
unsigned transactionId = commonCellUpdateConfirm(uep,ies);
ByteVector result(1000);
if (!encodeDcchMsg(uep,SRB2,&msg,result,descrCellUpdateConfirm)) {return;}
UeTransaction(uep,UeTransaction::ttCellUpdateConfirm, 0, transactionId);
uep->ueWriteHighSide(SRB2, result, descrCellUpdateConfirm);
}
static void sendCellUpdateConfirmCcch(UEInfo *uep)
{
ASN::DL_CCCH_Message_t msg;
memset(&msg,0,sizeof(msg));
msg.message.present = ASN::DL_CCCH_MessageType_PR_cellUpdateConfirm;
msg.message.choice.cellUpdateConfirm.present = ASN::CellUpdateConfirm_CCCH_PR_r3;
// U_RNTI_t u_RNTI;
toAsnURNTI(&msg.message.choice.cellUpdateConfirm.choice.r3.u_RNTI,uep->getSrncId(),uep->getSRNTI());
// CellUpdateConfirm_r3_IEs_t cellUpdateConfirm_r3;
ASN::CellUpdateConfirm_r3_IEs_t *ies =
&msg.message.choice.cellUpdateConfirm.choice.r3.cellUpdateConfirm_r3;
unsigned transactionId = commonCellUpdateConfirm(uep,ies);
ByteVector result(1000);
if (!encodeCcchMsg(&msg,result,descrCellUpdateConfirm,uep,0)) {return;}
UeTransaction(uep,UeTransaction::ttCellUpdateConfirm, 0, transactionId);
gMacSwitch.writeHighSideCcch(result,descrCellUpdateConfirm);
}
void sendCellUpdateConfirm(UEInfo *uep)
{
switch (uep->ueGetState()) {
case stCELL_DCH:
sendCellUpdateConfirmDcch(uep);
return;
default:
sendCellUpdateConfirmCcch(uep);
return;
}
}
// 33.102 5.1.2 Specifies two types of Security procedure: section 6.3 describes the main
// authentication method using Ki; section 6.5 describes a local authentication mechanism
// using an integrity key. You must do one of the two procedures at each connection setup,
// including the one for an L3 Service Request.
void sendSecurityModeCommand(UEInfo *uep)
{
ASN::DL_DCCH_Message_t msg;
memset(&msg,0,sizeof(msg));
msg.message.present = ASN::DL_DCCH_MessageType_PR_securityModeCommand;
msg.message.choice.securityModeCommand.present = ASN::SecurityModeCommand_PR_r3;
//ASN::SecurityModeCommand_t::SecurityModeCommand_u::SecurityMode_command__r3::Security *ies =
ASN::SecurityModeCommand_r3_IEs *ies =
&msg.message.choice.securityModeCommand.choice.r3.securityModeCommand_r3;
unsigned transactionId = uep->newTransactionId();
ies->rrc_TransactionIdentifier = transactionId;
// SecurityCapability_t securityCapability;
if (uep->radioCapability) {
// just copy this from UE Info to make UE happy.
// 8.1.12.3 implies that if these capabilities don't exactly match those of the UE's, the UE will release
ies->securityCapability = uep->radioCapability->securityCapability;
}
else {
ies->securityCapability.cipheringAlgorithmCap = allocAsnBIT_STRING(16);
// We are not going to do ciphering, but I am going to set one of the
// ciphering algorithm bits in case it is needed to make the UE happy.
AsnBitString2BVTemp(ies->securityCapability.cipheringAlgorithmCap).setField(
ASN::SecurityCapability__cipheringAlgorithmCap_uea0,1,1);
AsnBitString2BVTemp(ies->securityCapability.cipheringAlgorithmCap).setField(
ASN::SecurityCapability__cipheringAlgorithmCap_uea1,1,1);
ies->securityCapability.integrityProtectionAlgorithmCap = allocAsnBIT_STRING(16);
// Set the bit for algorithm UIA1, which is the kasumi-based one we support.
AsnBitString2BVTemp(ies->securityCapability.integrityProtectionAlgorithmCap).setField(
ASN::SecurityCapability__integrityProtectionAlgorithmCap_uia1,1,1); // FIXME: 10.3.3.7 of 25.331 doesn't follow this
}
// Not doing ciphering
// struct CipheringModeInfo *cipheringModeInfo /* OPTIONAL */;
/*ASN::CipheringModeInfo *cmi = RN_CALLOC(ASN::CipheringModeInfo);
ies->cipheringModeInfo = cmi;
cmi->cipheringModeCommand.present = ASN::CipheringModeCommand_PR_startRestart;
cmi->cipheringModeCommand.choice.startRestart = toAsnEnumerated(ASN::CipheringAlgorithm_uea0);
*/
// Since we are setting up Integrity Protection (and not just ciphering)
// we need to include this optional IE.
// struct IntegrityProtectionModeInfo *integrityProtectionModeInfo /* OPTIONAL */;
ASN::IntegrityProtectionModeInfo *ipmi = RN_CALLOC(ASN::IntegrityProtectionModeInfo);
ies->integrityProtectionModeInfo = ipmi;
ipmi->integrityProtectionModeCommand.present = ASN::IntegrityProtectionModeCommand_PR_startIntegrityProtection;
ipmi->integrityProtectionModeCommand.choice.startIntegrityProtection.integrityProtInitNumber = allocAsnBIT_STRING(32);
AsnBitString2BVTemp(ipmi->integrityProtectionModeCommand.choice.startIntegrityProtection.integrityProtInitNumber)
.setField(0, uep->integrity.getFresh(),32);
// This is optional - is it necessary?
ipmi->integrityProtectionAlgorithm = RN_CALLOC(ASN::IntegrityProtectionAlgorithm_t);
*ipmi->integrityProtectionAlgorithm = toAsnEnumerated(ASN::IntegrityProtectionAlgorithm_uia1);
ies->cn_DomainIdentity = toAsnEnumerated(ASN::CN_DomainIdentity_ps_domain);
// struct InterRAT_UE_SecurityCapList *ue_SystemSpecificSecurityCap /* OPTIONAL */;
/*ies->ue_SystemSpecificSecurityCap = RN_CALLOC(ASN::InterRAT_UE_SecurityCapList);
ASN::InterRAT_UE_SecurityCapability* secCap = RN_CALLOC(ASN::InterRAT_UE_SecurityCapability);
secCap->present = ASN::InterRAT_UE_SecurityCapability_PR_gsm;
secCap->choice.gsm.gsmSecurityCapability = allocAsnBIT_STRING(7);
AsnBitString2BVTemp(secCap->choice.gsm.gsmSecurityCapability).setField(ASN::GsmSecurityCapability_a5_1,1,1);
ASN_SEQUENCE_ADD(&ies->ue_SystemSpecificSecurityCap->list,secCap);
*/
// 25.331 8.5.10 and 8.6.3.5
// The security mode command itself is the first one that is protected.
// The start of the protection is indicated to the UE by the inclusion of the IntegrityCheck IE in the message.
// 13.4.10: Integrity Protection is turned off when entering/leaving idle mode.
uep->integrity.integrityStart();
ByteVector result(1000);
if (!encodeDcchMsg(uep,SRB2,&msg,result,descrSecurityModeCommand)) {return;}
UeTransaction(uep,UeTransaction::ttSecurityModeCommand, 0, transactionId);
uep->ueWriteHighSide(SRB2, result, descrSecurityModeCommand);
}
static void handleSecurityModeComplete(UEInfo*uep, ASN::SecurityModeComplete_t *secmsg)
{
// I'm not sure what to do with any of the optional contents of this message.
// struct IntegrityProtActivationInfo *ul_IntegProtActivationInfo /* OPTIONAL */;
// struct RB_ActivationTimeInfoList *rb_UL_CiphActivationTimeInfo /* OPTIONAL */;
// If the UE is not confused, the security procedure was initiated
// by the SGSN during an AttachRequest, so notify the SGSN that we are ready to roll.
uep->sgsnHandleSecurityModeComplete(true);
}
// Just print an error message based on the error code.
// We have lost communication with this UE and should do something.
// TODO: what?
static void handleSecurityModeFailure(UEInfo*uep, ASN::SecurityModeFailure_t *secfailmsg)
{
const char *why = "";
char whybuf[100];
switch (secfailmsg->failureCause.present) {
case ASN::FailureCauseWithProtErr_PR_configurationUnsupported: why = "configurationUnsupported"; break;
case ASN::FailureCauseWithProtErr_PR_physicalChannelFailure: why = "physicalChannelFailure";break;
case ASN::FailureCauseWithProtErr_PR_incompatibleSimultaneousReconfiguration: why="incompatibleSimultaneousReconfiguration"; break;
case ASN::FailureCauseWithProtErr_PR_compressedModeRuntimeError: why="compressedModeRuntimeError"; break;
case ASN::FailureCauseWithProtErr_PR_protocolError: {
why="protocolError";
// There is more interesting information provided with this cause:
ASN::ProtocolErrorInformation *pei = &secfailmsg->failureCause.choice.protocolError;
if (pei->diagnosticsType.present == ASN::ProtocolErrorInformation__diagnosticsType_PR_type1) {
long cause = asnEnum2long(pei->diagnosticsType.choice.type1.protocolErrorCause);
switch (cause) {
case ASN::ProtocolErrorCause_asn1_ViolationOrEncodingError:
why = "ProtocolErrorCause_asn1_ViolationOrEncodingError";
break;
case ASN::ProtocolErrorCause_messageTypeNonexistent:
why = "ProtocolErrorCause_messageTypeNonexistent";
break;
case ASN::ProtocolErrorCause_messageNotCompatibleWithReceiverState:
why = "ProtocolErrorCause_messageNotCompatibleWithReceiverState";
break;
case ASN::ProtocolErrorCause_ie_ValueNotComprehended:
why = "ProtocolErrorCause_ie_ValueNotComprehended";
break;
case ASN::ProtocolErrorCause_informationElementMissing:
why = "ProtocolErrorCause_informationElementMissing";
break;
case ASN::ProtocolErrorCause_messageExtensionNotComprehended:
why = "ProtocolErrorCause_messageExtensionNotComprehended";
break;
default:
sprintf(whybuf,"protocol error cause %ld",cause);
why = whybuf;
break;
}
}
break;
}
case ASN::FailureCauseWithProtErr_PR_cellUpdateOccurred: why="cellUpdateOccurred";break;
case ASN::FailureCauseWithProtErr_PR_invalidConfiguration: why="invalidConfiguration";break;
case ASN::FailureCauseWithProtErr_PR_configurationIncomplete: why="configurationIncomplete";break;
case ASN::FailureCauseWithProtErr_PR_unsupportedMeasurement: why="unsupportedMeasurement";break;
case ASN::FailureCauseWithProtErr_PR_mbmsSessionAlreadyReceivedCorrectly: why="mbmsSessionAlreadyRecievedCorrectly";break;
case ASN::FailureCauseWithProtErr_PR_lowerPriorityMBMSService: why="lowerPriorityMBMSService";break;
default: break;
}
LOG(ERR)<<format("Security Mode Failure cause=%d %s",secfailmsg->failureCause.present,why)<<uep;
uep->sgsnHandleSecurityModeComplete(false);
}
static void handleCellUpdate(ASN::CellUpdate_t *msg)
{
// This message sends a whole bunch of possible errors. Lets print them out.
bool srbErr = msg->am_RLC_ErrorIndicationRb2_3or4;
bool rbErr = msg->am_RLC_ErrorIndicationRb5orAbove;
bool radioErr = 0;
bool otherErr = 0;
long cuCause = 0;
long failCause = 0;
bool protoErrValid = 0; long protoErr;
// I think the transaction id in this message, which is hidden down in
// the failureCause, is the transaction id of the transaction that failed.
unsigned failedTransactionId = 0;
asn_INTEGER2long(&msg->cellUpdateCause,&cuCause);
switch (cuCause) {
// These are the two failure causes; all the other causes are normal.
case ASN::CellUpdateCause_radiolinkFailure: radioErr = true; break;
case ASN::CellUpdateCause_rlc_unrecoverableError: otherErr = true; break;
}
if (msg->failureCause) {
failedTransactionId = msg->failureCause->rrc_TransactionIdentifier;
failCause = msg->failureCause->failureCause.present;
if (failCause == ASN::FailureCauseWithProtErr_PR_protocolError) {
protoErrValid = true;
// Geesh. Just assume type1 and get it.
asn_INTEGER2long(&msg->failureCause->failureCause.choice.protocolError.
diagnosticsType.choice.type1.protocolErrorCause,&protoErr);
}
}
// They transmit the URNTI in its constituent SRNCid and S-RNTI parts.
uint32_t srnc = AsnBitString2BVTemp(msg->u_RNTI.srnc_Identity).getField(0,12);
uint32_t srnti = AsnBitString2BVTemp(msg->u_RNTI.s_RNTI).getField(0,20);
uint32_t urnti = (srnc<<20) | srnti;
const char *comment = "UL_CCCH_MessageType_PR_cellUpdate";
UEInfo *uep = gRrc.findUeByUrnti(urnti);
if (!uep) { comment = "UL_CCCH_MessageType_PR_cellUpdate (new UE)"; }
asnLogMsg(0, &ASN::asn_DEF_CellUpdate, msg,comment,uep,urnti);
bool anyErr = (srbErr || rbErr || otherErr || radioErr || failCause || protoErrValid);
if (anyErr) {
LOG(ERR) << "Received cell update message with failure info:"
<<LOGVAR(cuCause) <<LOGVAR(srbErr) <<LOGVAR(rbErr)
<<LOGVAR(otherErr) <<LOGVAR(radioErr) <<LOGVAR(failCause)
<<LOGVAR(protoErrValid) <<LOGVAR(protoErr)
<<LOGVAR(failedTransactionId)
<<uep;
} else if (uep == NULL) {
LOG(ERR) << format("Received cell update message with unrecognized URNTI=0x%x",
urnti);
}
if (uep) {
//msg->startList.
//if (rbscmsg->start_Value) {
// uint32_t startVal = AsnBitString2BV(rbscmsg->start_Value).getField(0,20);
// uep->integrity.updateStartValue(startVal);
//}
// Dont need this printf. The START value is not supposed to change. It was a bug that was changing it.
// printf("actual START value is: %u\n", uep->integrity.getStart());
sendCellUpdateConfirm(uep);
}
else {
// This is the first we have heard from this UE. This is illegal.
sendRrcConnectionReleaseCcch(urnti);
}
}
void handleRrcConnectionRequest(ASN::RRCConnectionRequest_t *msg)
{
//RrcUeId ueid(&msg->initialUE_Identity);
// establishmentCause; Do we care?
// We dont care about protocolError.
// We dont yet care about MeasuredResultsOnRACH.
// TODO: Do we need to worry about multiple requests from the same UE?
// If we get an identical request keep the same URNTI.
// (Not doing so was one of the bugs in GPRS.)
// There is an argument that if it is a duplicate request, just issue another URNTI,
// and whichever URNTI the UE decides to use will be fine with us, but I think
// that confuses the UE sometimes.
AsnUeId aid(msg->initialUE_Identity);
const char *comment = "UL_CCCH_MessageType_PR_rrcConnectionRequest";
UEInfo *uep = gRrc.findUeByAsnId(&aid);
if (uep == NULL) {
uep = new UEInfo(&aid);
comment = "UL_CCCH_MessageType_PR_rrcConnectionRequest (new UE)";
}
asnLogMsg(0, &ASN::asn_DEF_RRCConnectionRequest, msg,comment,uep);
// (pat 12-18-2012) If we get this it means the UE dropped into idle mode possibly
// without telling us, so fix that:
// FIXME: What about CELL_PCH or CELL_URA
uep->ueSetState(stIdleMode);
// We need to do stop integrity protection before sending the L3 Authentication message,
// which gets wrapped in an RRC direct transfer, which must not be integrity protected.
uep->integrity.integrityStop(); // Redudant with code in ueSetState, but make sure.
// Send the RRC Connection Setup Message,
// using the exact initialUE_Identity the UE provided, whatever it is.
sendRrcConnectionSetup(uep,&msg->initialUE_Identity);
}
#if 0 // Probably works, but not used, so take out to shut up compiler.
static bool fromAsnCnDomainIdentity(const char *inform, ASN::CN_DomainIdentity_t &asnthing)
{
bool psDomain;
long cnDomainId;
if (asn_INTEGER2long(&asnthing,&cnDomainId)) {
LOG(ERR) <<inform <<":"<<"invalid CN Domain Identity";
}
switch (cnDomainId) {
case ASN::CN_DomainIdentity_cs_domain: psDomain = false; break;
case ASN::CN_DomainIdentity_ps_domain: psDomain = true; break;
default: LOG(ERR) <<inform <<":"<<"invalid CN domain value";
}
return psDomain;
}
#endif
// This message is passed transparently from the MAC using RLC-TM.
// The MAC should pop off the headers and pass the rest.
// The only possible messages are:
// RRC Connection Request
// Cell Update
// URA Update
void rrcRecvCcchMessage(BitVector &tb,unsigned asc)
{
ASN::UL_CCCH_Message *msg1 = (ASN::UL_CCCH_Message*)uperDecodeFromBitV(&ASN::asn_DEF_UL_CCCH_Message, tb);
if (!msg1) {
LOG(ALERT) << "undecodable CCCH message received";
return;
}
//if (gConfig.getNum("UMTS.Debug.Messages")) {
// asn_fprint(stdout,&ASN::asn_DEF_UL_CCCH_Message, msg1);
// fflush(stdout);
//}
switch (msg1->message.present) {
case ASN::UL_CCCH_MessageType_PR_cellUpdate:
LOG(INFO) << "CELL_UPDATE message received";
//asn_fprint(stdout,&ASN::asn_DEF_CellUpdate, &msg1->message.choice.cellUpdate);
handleCellUpdate(&msg1->message.choice.cellUpdate);
return;
case ASN::UL_CCCH_MessageType_PR_rrcConnectionRequest:
LOG(INFO) << "RRC Connection Request received";
handleRrcConnectionRequest(&msg1->message.choice.rrcConnectionRequest);
return;
case ASN::UL_CCCH_MessageType_PR_uraUpdate:
// We didnt crack out the URNTI for the message so just leave it null.
asnLogMsg(0, &ASN::asn_DEF_UL_CCCH_Message, msg1,asnUlCcchMsg2Name(msg1->message.present),NULL);
LOG(INFO) << "URA_UPDATE message ignored";
return;
default:
asnLogMsg(0, &ASN::asn_DEF_UL_CCCH_Message, msg1,asnUlCcchMsg2Name(msg1->message.present),NULL);
LOG(ERR) << "CCCH message ignored, unknown type="<<msg1->message.present;
return;
}
ASN_STRUCT_FREE(ASN::asn_DEF_UL_CCCH_Message, msg1);
}
void UEInfo::ueRecvL3Msg(ByteVector &msgframe, UEInfo *uep)
{
unsigned pd = msgframe.getNibble(0,0); // protocol descriminator
LOG(INFO) << "Received L3 message of with protocol discriminator " << pd;
switch ((GSM::L3PD) pd) {
case GSM::L3GPRSMobilityManagementPD: // Couldnt we shorten this?
case GSM::L3GPRSSessionManagementPD: // Couldnt we shorten this?
//LOG(INFO) << "Sending L3 message of descr " << pd << "up to SGSN";
sgsnHandleL3Msg(uep->mURNTI,msgframe);
//LOG(INFO) << "Sent to SGSN";
break;
// TODO: Send GSM messages somewhere
case GSM::L3CallControlPD:
case GSM::L3MobilityManagementPD:
case GSM::L3RadioResourcePD:
// In GSM these go on the logical channel, which is polled by DCCHDispatcher,
// then calls DCCHDispatchMessage, which then calls some sub-processor
// which may generate message traffic on a LogicalChannel class.
// The best way to interface to the existing code is probably to put
// these on the LogicalChannel and let the DCCH service loop find them.
// It wants to find an ESTABLISH primitive as the first thing,
// and then it times out if nothing happens soon?
// Or we could try calling direct: DCCHDispatchMessage(??,??);
// FIXME: Ignore these until L3 GSM code is integrated
LOG(ERR) << "L3 GSM control message ignored";
//uep->mGsmL3->l3writeHighSide(msgframe);
return;
case GSM::L3SMSPD:
// In GSM these apparently arrive on the DTCHLogicalChannel?
// Not sure how SMS works. Looks like an MM message CMServiceRequest
// arrives to DCCHDispatchMM() which then calls CMServiceResponder()
// which calls MOSMSController() which seems to do a bunch of message
// traffic on a DCCHLogicalChannel, and SMS messages go there.
LOG(ERR) <<"L3 SMS Message ignored";
return;
default:
LOG(ERR)<< "unsupported L3 Message PD:"<<pd;
}
}
void UEInfo::ueRecvStatusMsg(ASN::UL_DCCH_Message *msg1)
{
ASN::RRCStatus_t *statusmsg = &msg1->message.choice.rrcStatus;
if (statusmsg->protocolErrorInformation.diagnosticsType.present !=
ASN::ProtocolErrorMoreInformation__diagnosticsType_PR_type1) {
LOG(WARNING) <<"Received RRCStatus that is not type1 " <<this;
return; // Cant do anything more; type1 is all we handle.
}
// ASN gets a little verbose here...
ASN::ProtocolErrorMoreInformation::
ProtocolErrorMoreInformation__diagnosticsType::
ProtocolErrorMoreInformation__diagnosticsType_u::
ProtocolErrorMoreInformation__diagnosticsType__type1 *info =
&statusmsg->protocolErrorInformation.diagnosticsType.choice.type1;
char const *errtype;
#define STATUS_INFO_TYPE(what) \
ASN::ProtocolErrorMoreInformation__diagnosticsType__type1_PR_##what: \
errtype = #what;
// This is a overkill since rmsg is in the same location in the union every time.
ASN::IdentificationOfReceivedMessage_t *rmsg = 0;
switch (info->present) {
case STATUS_INFO_TYPE(asn1_ViolationOrEncodingError)
break;
case STATUS_INFO_TYPE(messageTypeNonexistent)
break;
case STATUS_INFO_TYPE(messageNotCompatibleWithReceiverState)
rmsg = &info->choice.messageNotCompatibleWithReceiverState;
break;
case STATUS_INFO_TYPE(ie_ValueNotComprehended)
rmsg = &info->choice.ie_ValueNotComprehended;
break;
case STATUS_INFO_TYPE(conditionalInformationElementError)
rmsg = &info->choice.conditionalInformationElementError;
break;
case STATUS_INFO_TYPE(messageExtensionNotComprehended)
rmsg = &info->choice.messageExtensionNotComprehended;
break;
default: errtype = "unrecognized";
}
// Finally, lets print something.
// I'm not going to bother decoding all this info into strings,
// so the message wont make sense unless you have access to ASN and our code.
char rbuf[200]; rbuf[0] = 0;
if (rmsg) {
long rmsgtype;
asn_INTEGER2long(&rmsg->receivedMessageType,&rmsgtype);
unsigned transId = rmsg->rrc_TransactionIdentifier;
UeTransaction *tr = this->getTransactionRaw(transId,"RrcStatus");
sprintf(rbuf," ReceivedMessageType=%ld transId=%d %s",
rmsgtype,transId,tr?tr->str().c_str():"");
}
LOG(WARNING) <<"Received RRCStatus errnum="<<(int)info->present
<<" errtype="<<errtype <<this <<rbuf;
}
// This is called by the high side of the RLC engine for SRB1,2,3,4.
void UEInfo::ueRecvDcchMessage(ByteVector &bv,unsigned rbNum)
{
ueRegisterActivity(); // Not sure if all these messages count as activity.
UEInfo *uep = this;
//bool verbose = gConfig.getNum("UMTS.Debug.Messages"); // redundant with asnLogMsg.
//if (verbose) { fprintf(stdout,"Received DCCH message for %s rb=%d\n",uep->ueid().c_str(),rbNum); }
LOG(INFO) << "DCCH ByteVector: " << bv;
ASN::UL_DCCH_Message *msg1 = (ASN::UL_DCCH_Message*)uperDecodeFromByteV(&ASN::asn_DEF_UL_DCCH_Message, bv);
if (!msg1) return;
//if (verbose) { asn_fprint(stdout,&ASN::asn_DEF_UL_DCCH_Message, msg1); fflush(stdout); }
asnLogMsg(rbNum, &ASN::asn_DEF_UL_DCCH_Message, msg1, asnUlDcchMsg2Name(msg1->message.present),uep);
//const char *inform; // informative string for error messages.
unsigned transId; // Transaction Id.
// The DCCH messages should arrive on rbnum 3, but who cares.
switch (msg1->message.present) {
case ASN::UL_DCCH_MessageType_PR_rrcConnectionSetupComplete: {
ASN::RRCConnectionSetupComplete *rcscmsg =
&msg1->message.choice.rrcConnectionSetupComplete;
if (rcscmsg->ue_RadioAccessCapability) {
uep->radioCapability = RN_CALLOC(ASN::UE_RadioAccessCapability);
memcpy(uep->radioCapability,rcscmsg->ue_RadioAccessCapability,sizeof(struct ASN::UE_RadioAccessCapability));
}
// This message should arrive on SRB1 to indicate initial RRC connection is set up.
// If you get this far, you have succeeded in establishing an RRC connection.
LOG(NOTICE) << "Received RRC Connection Setup Complete! Congratulations!";
// The fact of the message arriving is the major piece of information.
// Message also contains START info and UE Radio Access Capability Info.
// The UE has already switched to CELL_FACH state in order to send this
// message on DCCH.
// Now inform ourselves; we already connected DCCH appropriately
// before sending the ConnectionSetup msg.
// This transaction was handled specially in ueWriteLowSide because
// we had to process it before it can go through the RLC.
//transId = rcscmsg->rrc_TransactionIdentifier;
//uep->ueRecvRrcConnectionSetupResponse(transId,true,"RrcConnectionSetupComplete");
break;
}
case ASN::UL_DCCH_MessageType_PR_rrcConnectionReleaseComplete: {
ASN::RRCConnectionReleaseComplete *rcrcmsg =
&msg1->message.choice.rrcConnectionReleaseComplete;
transId = rcrcmsg->rrc_TransactionIdentifier;
uep->ueRecvRrcConnectionReleaseResponse(transId,true,"RrcConnectionReleaseComplete");
break;
}
case ASN::UL_DCCH_MessageType_PR_radioBearerSetupComplete: {
// Currently we only send RadioBearerSetup for PS connections, so it must be for an SGSN PDP context.
ASN::RadioBearerSetupComplete *rbscmsg =
&msg1->message.choice.radioBearerSetupComplete;
// I am ignoring the transaction identifier.
// I am ignoring the "START" values.
///*if (rbscmsg->start_Value) {
// uint32_t startVal = AsnBitString2BV(rbscmsg->start_Value).getField(0,20);
// uep->integrity.updateStartValue(startVal);
//}*/
// We need the rbid of the PDP context, which in the RRCP spec is in
// "RB with PDPCP Information", but this does not appear in our ASN description,
// unless it is buried inside the extensions which are a bitstring,
// and no decoder provided.
// So I guess the presense of this message and the transactionId
// is the only indication we get.
transId = rbscmsg->rrc_TransactionIdentifier;
uep->ueRecvRadioBearerSetupResponse(transId,true,"RadioBearerSetupComplete");
break;
}
case ASN::UL_DCCH_MessageType_PR_radioBearerSetupFailure: {
ASN::RadioBearerSetupFailure *rbsfmsg =
&msg1->message.choice.radioBearerSetupFailure;
transId = rbsfmsg->rrc_TransactionIdentifier;
// Print out the failure cause; Just print the ASN enum value directly.
// It is FailureCauseWithProtErr_t
unsigned failcode = rbsfmsg->failureCause.present;
string msg = getRBFailureCause(&rbsfmsg->failureCause);
LOG(ERR) << "Radio Bearer Setup Failure code="<<failcode <<" ("<<msg<<")"
<<" rb="<<rbNum
<<" ue:"<<uep;
uep->ueRecvRadioBearerSetupResponse(transId,false,"RadioBearerSetupFailure");
break;
}
// TODO: Not sure we need these at all:
case ASN::UL_DCCH_MessageType_PR_radioBearerReleaseComplete: {
ASN::RadioBearerReleaseComplete *rbrcmsg =
&msg1->message.choice.radioBearerReleaseComplete;
transId = rbrcmsg->rrc_TransactionIdentifier;
uep->ueRecvRadioBearerReleaseResponse(transId,true,"RadioBearerReleaseComplete");
break;
}
case ASN::UL_DCCH_MessageType_PR_radioBearerReleaseFailure: {
ASN::RadioBearerReleaseFailure *rbrfmsg =
&msg1->message.choice.radioBearerReleaseFailure;
transId = rbrfmsg->rrc_TransactionIdentifier;
// Print out the failure cause; Just print the ASN enum value directly.
// It is FailureCauseWIthProtErr_t
unsigned failcode = rbrfmsg->failureCause.present;
LOG(ERR) << "Radio Bearer Setup Failure code="<<failcode
<<" rb="<<rbNum
<<" ue:"<<uep;
uep->ueRecvRadioBearerReleaseResponse(transId,false,"RadioBearerReleaseFailure");
break;
}
case ASN::UL_DCCH_MessageType_PR_uplinkDirectTransfer: {
// This message should arrive on SRB3
// This is a subsequent uplink message to L3.
//inform = "Uplink Direct Transfer Message";
ASN::UplinkDirectTransfer_t *udtmsg = &msg1->message.choice.uplinkDirectTransfer;
// Do we care about this ps-cs domain flag at this point?
// We can direct the message based on the message protocol descriptor.
// bool psDomain = fromAsnCnDomainIdentity(&udtmsg.cn_DomainIdentity);
ByteVector l3UdtMsgBytes(udtmsg->nas_Message.buf,udtmsg->nas_Message.size);
uep->ueRecvL3Msg(l3UdtMsgBytes,uep);
break;
}
case ASN::UL_DCCH_MessageType_PR_initialDirectTransfer: {
//inform = "Initial Direct Transfer Message";
// This message should arrive on SRB3
// This is the first uplink message to L3.
// On the UE side, it causes SRB1-SRB3 (and optionally SRB4) to be set up
// before it complete. On the network side, it is like the uplink direct transfer
// but includes setup information for routing on the CN [Core Network] side.
// This msg contains IntraDomainNasNodeSelector which has another UE-id in it,
// but it doesnt include anything really interesting, for example, if it is an IMSI,
// they only send a few bits of it, not the whole thing,
// we already know what UE we are talking to, so dont bother to even look at it.
ASN::InitialDirectTransfer_t *idtmsg = &msg1->message.choice.initialDirectTransfer;
// Get CS or PS?
// bool psDomain = fromAsnCnDomainIdentity(&idtmsg.cn_DomainIdentity);
// We can direct the message based on the message protocol descriptor.
// What we want is the NAS message, which is an ASN::OCTET_STRING;
// convert it to an allocated ByteVector; allocate it just for safety's sake,
// but TODO: Use a ByteVectorTemp because we shouldnt need to allocate it,
// we can just use it right out of the OCTET_STRING struct.
ByteVector l3IdtMsgBytes(idtmsg->nas_Message.buf,idtmsg->nas_Message.size);
uep->ueRecvL3Msg(l3IdtMsgBytes,uep);
break;
}
case ASN::UL_DCCH_MessageType_PR_rrcStatus: {
// This sends an error indication about some received message.
uep->ueRecvStatusMsg(msg1);
break;
}
case ASN::UL_DCCH_MessageType_PR_signallingConnectionReleaseIndication: {
// The UE sends this to tell us it wants to drop the connection.
// FIXME: Or does it send this to tell us it already dropped the connection? In which case
// we dont need to send the RrcConnectionRelease message.
// Harvind (3-17-13). This message is the industry standard for terminating a DCH connection to save battery life
// SCRI->RRC Conn. Release
// When UE wants to reconnect... RRC Conn. Setup->Service Request procedure
UeTransaction *lastTrans = getLastTransaction();
if (lastTrans->mTransactionType != ttComplete && lastTrans->elapsed()<1000) {
// There is some current transaction happening.
// Lets wait a while to see what happens.
return;
}
switch (uep->ueGetState()) {
case stIdleMode:
// This is not possible.
LOG(ERR) << "Received SignallingConnectionReleaseIndication message"
<<" for ue inIdleMode"<<uep;
break;
case stCELL_DCH:
// The SGSN will call back to rrc to send a RadioBearerRelease message,
// which should cause the UE to send RadioBearerReleaseComplete/Failure,
// which will kick the UE down to CELL_FACH state.
// Harvind (3-17-13) See above. UE barfs when we send RadioBearerRelease.
uep->ueSetState(stCELL_DCH);
if (lastTrans->mTransactionType == ttRadioBearerRelease &&
lastTrans->elapsed()<1000) {
// Hmm. Its not working. Not sure what to do. Try harder.
sendRrcConnectionRelease(uep);
} else {
// Harvind (3-17-13) See above. Keep PDP context intact for smoother operation.
//printf("Freeing all PDP contexts.");
//uep->sgsnFreePdpAll(mURNTI);
sendRrcConnectionRelease(uep);
}
break;
default:
sendRrcConnectionRelease(uep);
break;
}
break;
}
case ASN::UL_DCCH_MessageType_PR_securityModeComplete: {
ASN::SecurityModeComplete_t *secmsg = &msg1->message.choice.securityModeComplete;
transId = secmsg->rrc_TransactionIdentifier;
handleSecurityModeComplete(uep,secmsg);
UeTransaction *tr = getTransaction(transId,ttSecurityModeCommand,"SecurityModeComplete");
// The security mode was started when we sent the command, not when we receive the response,
// so there is nothing special to do here. If we changed the Kc by re-running the Layer3
// Authentication procedure while a connection was running, then we would have to apply the
// new keys at this time, but we dont ever do that.
if (tr) tr->transClose(); // Done with this one.
break;
}
case ASN::UL_DCCH_MessageType_PR_securityModeFailure: {
ASN::SecurityModeFailure_t *secfailmsg = &msg1->message.choice.securityModeFailure;
transId = secfailmsg->rrc_TransactionIdentifier;
handleSecurityModeFailure(uep,secfailmsg);
UeTransaction *tr = getTransaction(transId,ttSecurityModeCommand,"SecurityModeFailure");
if (tr) tr->transClose(); // Done with this one.
break;
}
default:
LOG(ERR) << "DCCH message ignored, unhandled type="<<msg1->message.present
<<" "<<asnUlDcchMsg2Name(msg1->message.present);
}
}
bool AsnUeId::eql(AsnUeId &other)
{
// All fields are inited, so just compare everything.
if (idType != other.idType) return false;
if (mImsi != other.mImsi || mImei != other.mImei || mTmsiDS41 != other.mTmsiDS41) return false;
if (mMcc != other.mMcc || mMnc != other.mMnc) return false;
if (mTmsi != other.mTmsi || mPtmsi != other.mPtmsi || mEsn != other.mEsn) return false;
if (mLac != other.mLac || mRac != other.mRac) return false;
return true;
}
void AsnUeId::asnParse(ASN::InitialUE_Identity &uid)
{
switch (uid.present) {
case ASN::InitialUE_Identity_PR_imsi:
// A_SEQUENCE_OF(Digit_t) list
mImsi = AsnSeqOfDigit2BV(&uid.choice.imsi.list);
break;
case ASN::InitialUE_Identity_PR_tmsi_and_LAI:
// TMSI_GSM_MAP_t (aka BIT_STRING_t) tmsi
// LAI_t lai;
// PLMN_Identity_t plmn_Identity
// MCC_t (A_SEQUENCE_OF(Digit_t) mcc;
// MNC_t (A_SEQUENCE_OF(Digit_t) mnc;
// BIT_STRING_t lac;
mTmsi = AsnBitString2BVTemp(uid.choice.tmsi_and_LAI.tmsi).getField(0,32);
mLac = AsnBitString2BVTemp(uid.choice.tmsi_and_LAI.lai.lac).getField(0,16);
{ ByteVector bmcc = AsnSeqOfDigit2BV(&uid.choice.tmsi_and_LAI.lai.plmn_Identity.mcc);
mMcc = bmcc.getField(0,bmcc.sizeBits());
}
{ ByteVector bmnc = AsnSeqOfDigit2BV(&uid.choice.tmsi_and_LAI.lai.plmn_Identity.mnc);
mMnc = bmnc.getField(0,bmnc.sizeBits());
}
//mMcc = AsnSeqOfDigit2BV(&uid.choice.tmsi_and_LAI.lai.plmn_Identity.mcc);
//mMnc = AsnSeqOfDigit2BV(&uid.choice.tmsi_and_LAI.lai.plmn_Identity.mnc);
break;
case ASN::InitialUE_Identity_PR_p_TMSI_and_RAI:
// P_TMSI_GSM_MAP_t (aka BIT_STRING_t) p_TMSI;
// RAI_t rai;
// LAI_t lai;
// PLMN_Identity_t (see above) plmn_Identity;
// BIT_STRING_t lac;
// RoutingAreaCode_t (aka BIT_STRING_t) rac;
mPtmsi = AsnBitString2BVTemp(uid.choice.p_TMSI_and_RAI.p_TMSI).getField(0,32);
mLac = AsnBitString2BVTemp(uid.choice.p_TMSI_and_RAI.rai.lai.lac).getField(0,16);
mRac = AsnBitString2BVTemp(uid.choice.p_TMSI_and_RAI.rai.rac).getField(0,8);
{ ByteVector bmcc = AsnSeqOfDigit2BV(&uid.choice.p_TMSI_and_RAI.rai.lai.plmn_Identity.mcc);
mMcc = bmcc.getField(0,bmcc.sizeBits());
}
{ ByteVector bmnc = AsnSeqOfDigit2BV(&uid.choice.p_TMSI_and_RAI.rai.lai.plmn_Identity.mnc);
mMnc = bmnc.getField(0,bmnc.sizeBits());
}
//mMcc = AsnSeqOfDigit2BV(&uid.choice.p_TMSI_and_RAI.rai.lai.plmn_Identity.mcc);
//mMnc = AsnSeqOfDigit2BV(&uid.choice.p_TMSI_and_RAI.rai.lai.plmn_Identity.mnc);
break;
case ASN::InitialUE_Identity_PR_imei:
// A_SEQUENCE_OF(Digit_t) list
mImei =AsnSeqOfDigit2BV(&uid.choice.imei.list);
break;
case ASN::InitialUE_Identity_PR_esn_DS_41:
// BIT_STRING_t // 32 bits
mEsn = AsnBitString2BVTemp(uid.choice.esn_DS_41).getField(0,32);
break;
case ASN::InitialUE_Identity_PR_imsi_DS_41:
// OCTET_STRING_t
mImsi = AsnOctetString2BV(uid.choice.imsi_DS_41);
break;
case ASN::InitialUE_Identity_PR_imsi_and_ESN_DS_41:
// IMSI_DS_41_t (aka OCTET_STRING_t) imsi_DS_41; // 5-7 bytes
// ESN_DS_41_t (aka BIT_STRING_t) esn_DS_41; // 32 bits.
mImsi = AsnOctetString2BV(uid.choice.imsi_and_ESN_DS_41.imsi_DS_41);
mEsn = AsnBitString2BVTemp(uid.choice.imsi_and_ESN_DS_41.esn_DS_41).getField(0,32);
break;
case ASN::InitialUE_Identity_PR_tmsi_DS_41:
// OCTET_STRING_t
mTmsiDS41 = AsnOctetString2BV(uid.choice.tmsi_DS_41);
break;
default:
LOG(ERR) << "Unexpected UE id type:"<<(int)uid.present;
break;
}
}
#if 0 // Works, but unused.
// Get the MCC as a ByteVector. Assumes str is correct.
static ByteVector stringOfDigitsToByteVector(const string &str)
{
ByteVector result(str.length());
for (unsigned i=0; i < str.length(); i++) {
result.setByte(i,(int8_t) str[i] - '0');
}
return result;
}
#endif
// Does the routing/location area in this asn id match our BTS?
bool AsnUeId::RaiMatches()
{
// MCC and MNC from the PLMN_identity must exist and match.
//if (mMcc.size() == 0 || mMnc.size() == 0) { return false; }
const string mccstr = gConfig.getStr("UMTS.Identity.MCC");
const string mncstr = gConfig.getStr("UMTS.Identity.MNC");
unsigned mcc = atoi(mccstr.c_str());
unsigned mnc = atoi(mncstr.c_str());
//ByteVector mcc = stringOfDigitsToByteVector(mccstr);
//ByteVector mnc = stringOfDigitsToByteVector(mncstr);
if (mMcc != mcc && mMnc != mnc) {
LOG(INFO) << "RAI of UE"<<LOGVAR(mcc)<<LOGVAR(mnc) <<" does not match"<<LOGVAR(mMcc)<<LOGVAR(mMnc);
return false;
}
// LAC and RAC must match.
unsigned lac = gConfig.getNum("UMTS.Identity.LAC");
unsigned rac = gConfig.getNum("GPRS.RAC");
if (lac != mLac || rac != mRac) {
LOG(INFO) << "RAI of UE" <<LOGVAR(lac)<<LOGVAR(rac)<<" does not match"<<LOGVAR(mLac)<<LOGVAR(mRac);
return false;
}
LOG(INFO) << "RAI of UE matches";
return true;
}
}; // namespace UMTS