mirror of
https://github.com/RangeNetworks/openbts.git
synced 2025-10-23 16:13:52 +00:00
302 lines
7.0 KiB
C++
302 lines
7.0 KiB
C++
/**@file
|
|
@brief Common elements for L3 messages, GSM 04.08 10.5.1.
|
|
*/
|
|
/*
|
|
* Copyright 2008-2010 Free Software Foundation, Inc.
|
|
* Copyright 2010 Kestrel Signal Processing, Inc.
|
|
* Copyright 2014 Range Networks, Inc.
|
|
*
|
|
* This software is distributed under multiple licenses;
|
|
* see the COPYING file in the main directory for licensing
|
|
* information for this specific distribution.
|
|
*
|
|
* This use of this software may be subject to additional restrictions.
|
|
* See the LEGAL file in the main directory for details.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#ifndef GSMCOMMONELEMENTS_H
|
|
#define GSMCOMMONELEMENTS_H
|
|
|
|
#include "GSML3Message.h"
|
|
#include <ControlTransfer.h>
|
|
#include <OpenBTSConfig.h>
|
|
|
|
|
|
namespace GSM {
|
|
|
|
|
|
|
|
/** Cell Identity, GSM 04.08 10.5.1.1 */
|
|
class L3CellIdentity : public L3ProtocolElement {
|
|
|
|
private:
|
|
|
|
unsigned mID;
|
|
|
|
public:
|
|
|
|
L3CellIdentity(unsigned wID=gConfig.getNum("GSM.Identity.CI"))
|
|
:mID(wID)
|
|
{ }
|
|
|
|
size_t lengthV() const { return 2; }
|
|
void writeV(L3Frame& dest, size_t &wp) const;
|
|
void parseV(const L3Frame&, size_t&) { abort(); }
|
|
void parseV(const L3Frame&, size_t&, size_t) { abort(); }
|
|
void text(std::ostream&) const;
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/** LAI, GSM 04.08 10.5.1.3 */
|
|
class L3LocationAreaIdentity : public L3ProtocolElement {
|
|
|
|
private:
|
|
|
|
unsigned mMCC[3]; ///< mobile country code digits
|
|
unsigned mMNC[3]; ///< mobile network code digits
|
|
unsigned mLAC; ///< location area code
|
|
|
|
|
|
public:
|
|
|
|
/**
|
|
Initialize the LAI with real values, drawn from gConfig by default.
|
|
@param wMCC MCC as a string (3 digits).
|
|
@param wMNC MNC as a string (2 or 3 digits).
|
|
@param wLAC LAC as a number.
|
|
*/
|
|
L3LocationAreaIdentity(
|
|
const char*wMCC = gConfig.getStr("GSM.Identity.MCC").c_str(),
|
|
const char* wMNC = gConfig.getStr("GSM.Identity.MNC").c_str(),
|
|
unsigned wLAC = gConfig.getNum("GSM.Identity.LAC")
|
|
);
|
|
|
|
/** Sometimes we need to compare these things. */
|
|
bool operator==(const L3LocationAreaIdentity&) const;
|
|
|
|
size_t lengthV() const { return 5; }
|
|
void parseV(const L3Frame& source, size_t &rp);
|
|
void parseV(const L3Frame&, size_t&, size_t) { abort(); }
|
|
void writeV(L3Frame& dest, size_t &wp) const;
|
|
void text(std::ostream&) const;
|
|
|
|
int MCC() const { return mMCC[0]*100 + mMCC[1]*10 + mMCC[2]; }
|
|
int MNC() const;
|
|
int LAC() const { return mLAC; }
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/** Mobile Identity, GSM 04.08, 10.5.1.4 */
|
|
class L3MobileIdentity : public L3ProtocolElement {
|
|
private:
|
|
|
|
MobileIDType mType; ///< IMSI, TMSI, or IMEI?
|
|
char mDigits[16]; ///< GSM 03.03 2.2 limits the IMSI or IMEI to 15 digits.
|
|
uint32_t mTMSI; ///< GSM 03.03 2.4 specifies the TMSI as 32 bits
|
|
|
|
public:
|
|
|
|
/** Empty ID */
|
|
L3MobileIdentity()
|
|
:L3ProtocolElement(),
|
|
mType(NoIDType)
|
|
{ mDigits[0]='\0'; }
|
|
|
|
/** TMSI initializer. */
|
|
L3MobileIdentity(unsigned int wTMSI)
|
|
:L3ProtocolElement(),
|
|
mType(TMSIType), mTMSI(wTMSI)
|
|
{ mDigits[0]='\0'; }
|
|
|
|
/** IMSI initializer. */
|
|
L3MobileIdentity(const char* wDigits)
|
|
:L3ProtocolElement(),
|
|
mType(IMSIType)
|
|
{ assert(strlen(wDigits)<=15); strcpy(mDigits,wDigits); }
|
|
|
|
/**@name Accessors. */
|
|
//@{
|
|
MobileIDType type() const { return mType; }
|
|
const char* digits() const { assert(mType!=TMSIType); return mDigits; }
|
|
unsigned int TMSI() const { assert(mType==TMSIType); return mTMSI; }
|
|
bool isIMSI() const { return mType==IMSIType; }
|
|
bool isTMSI() const { return mType==TMSIType; }
|
|
//@}
|
|
|
|
/** Comparison. */
|
|
bool operator==(const L3MobileIdentity&) const;
|
|
|
|
bool fmidMatch(const Control::FullMobileId *mobileId) const;
|
|
|
|
/** Comparison. */
|
|
bool operator!=(const L3MobileIdentity& other) const { return !operator==(other); }
|
|
|
|
/** Comparison. */
|
|
bool operator<(const L3MobileIdentity&) const;
|
|
|
|
size_t lengthV() const;
|
|
void writeV(L3Frame& dest, size_t &wp) const;
|
|
void parseV( const L3Frame& src, size_t &rp, size_t expectedLength );
|
|
void parseV(const L3Frame&, size_t&) { abort(); }
|
|
void text(std::ostream&) const;
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
Mobile Station Classmark 1, GSM 04.08 10.5.1.5
|
|
*/
|
|
class L3MobileStationClassmark1 : public L3ProtocolElement {
|
|
|
|
protected:
|
|
|
|
unsigned mRevisionLevel;
|
|
unsigned mES_IND;
|
|
unsigned mA5_1;
|
|
unsigned mRFPowerCapability;
|
|
|
|
public:
|
|
|
|
size_t lengthV() const { return 1; }
|
|
void writeV(L3Frame&, size_t&) const { assert(0); }
|
|
void parseV(const L3Frame &src, size_t &rp);
|
|
void parseV(const L3Frame&, size_t&, size_t) { assert(0); }
|
|
void text(std::ostream&) const;
|
|
|
|
};
|
|
|
|
|
|
// (pat) Added 9-2014. Encoding for encryption algorithms.
|
|
struct EncryptionAlgorithm {
|
|
// These bits define a mask of supported encryption algorithms.
|
|
enum {
|
|
Bit5_1 = 1,
|
|
Bit5_2 = 2,
|
|
Bit5_3 = 4
|
|
};
|
|
};
|
|
/**
|
|
Mobile Station Classmark 2, GSM 04.08 10.5.1.5
|
|
*/
|
|
class L3MobileStationClassmark2 : public L3ProtocolElement {
|
|
|
|
protected:
|
|
|
|
unsigned mRevisionLevel;
|
|
unsigned mES_IND;
|
|
unsigned mA5_1;
|
|
unsigned mA5_3;
|
|
unsigned mA5_2;
|
|
unsigned mRFPowerCapability;
|
|
unsigned mPSCapability;
|
|
unsigned mSSScreenIndicator;
|
|
unsigned mSMCapability;
|
|
unsigned mVBS;
|
|
unsigned mVGCS;
|
|
unsigned mFC;
|
|
unsigned mCM3;
|
|
unsigned mLCSVACapability;
|
|
unsigned mSoLSA;
|
|
unsigned mCMSF;
|
|
|
|
public:
|
|
|
|
size_t lengthV() const { return 3; }
|
|
void writeV(L3Frame&, size_t&) const { assert(0); }
|
|
void parseV(const L3Frame &src, size_t &rp);
|
|
void parseV(const L3Frame&, size_t&, size_t);
|
|
void text(std::ostream&) const;
|
|
|
|
// These return true if the encryption type is supported.
|
|
private:
|
|
bool A5_1() const { return mA5_1==0; }
|
|
bool A5_2() const { return mA5_2!=0; }
|
|
bool A5_3() const { return mA5_3!=0; }
|
|
public:
|
|
// (pat) Pre-9-2014 encoding: int getA5Bits() const { return (A5_1()<<2) + (A5_2()<<1) + A5_3(); }
|
|
int getA5Bits() const {
|
|
int result = 0;
|
|
if (A5_1()) { result |= EncryptionAlgorithm::Bit5_1; }
|
|
if (A5_2()) { result |= EncryptionAlgorithm::Bit5_2; }
|
|
if (A5_3()) { result |= EncryptionAlgorithm::Bit5_3; }
|
|
return result;
|
|
}
|
|
|
|
// Returns the power class, based on power capability encoding.
|
|
int powerClass() const { return mRFPowerCapability+1; }
|
|
};
|
|
|
|
|
|
/**
|
|
Mobile Station Classmark 3, GSM 04.08 10.5.1.7
|
|
FIXME -- We are only parsing the A5/x bits.
|
|
*/
|
|
class L3MobileStationClassmark3 : public L3ProtocolElement {
|
|
|
|
protected:
|
|
|
|
unsigned mMultiband;
|
|
unsigned mA5_4;
|
|
unsigned mA5_5;
|
|
unsigned mA5_6;
|
|
unsigned mA5_7;
|
|
|
|
public:
|
|
|
|
L3MobileStationClassmark3()
|
|
:mA5_4(0),mA5_5(0),mA5_6(0),mA5_7(0)
|
|
{ }
|
|
|
|
size_t lengthV() const { return 14; } // HACK -- Return maximum length.
|
|
void writeV(L3Frame&, size_t&) const { assert(0); }
|
|
void parseV(const L3Frame &, size_t &);
|
|
void parseV(const L3Frame&, size_t&, size_t);
|
|
void text(std::ostream&) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
class L3CipheringKeySequenceNumber : public L3ProtocolElement {
|
|
|
|
protected:
|
|
|
|
unsigned mCIValue;
|
|
|
|
public:
|
|
|
|
L3CipheringKeySequenceNumber(unsigned wCIValue):
|
|
mCIValue(wCIValue)
|
|
{ }
|
|
|
|
size_t lengthV() const { return 0; }
|
|
void writeV(L3Frame&, size_t&) const;
|
|
void parseV(const L3Frame &, size_t &) { assert(0); }
|
|
void parseV(const L3Frame&, size_t&, size_t) { assert(0); }
|
|
void text(std::ostream&) const;
|
|
};
|
|
|
|
|
|
|
|
|
|
} // GSM
|
|
|
|
#endif
|
|
|
|
// vim: ts=4 sw=4
|