#ifndef _GPRS_LLC_H #define _GPRS_LLC_H #include #include /* Section 4.7 LLC Layer Structure */ enum gprs_llc_sapi { GPRS_SAPI_GMM = 1, GPRS_SAPI_TOM2 = 2, GPRS_SAPI_SNDCP3 = 3, GPRS_SAPI_SNDCP5 = 5, GPRS_SAPI_SMS = 7, GPRS_SAPI_TOM8 = 8, GPRS_SAPI_SNDCP9 = 9, GPRS_SAPI_SNDCP11 = 11, }; /* Section 6.4 Commands and Responses */ enum gprs_llc_u_cmd { GPRS_LLC_U_DM_RESP = 0x01, GPRS_LLC_U_DISC_CMD = 0x04, GPRS_LLC_U_UA_RESP = 0x06, GPRS_LLC_U_SABM_CMD = 0x07, GPRS_LLC_U_FRMR_RESP = 0x08, GPRS_LLC_U_XID = 0x0b, GPRS_LLC_U_NULL_CMD = 0x00, }; /* TS 04.64 Section 7.1.2 Table 7: LLC layer primitives (GMM/SNDCP/SMS/TOM) */ /* TS 04.65 Section 5.1.2 Table 2: Service primitives used by SNDCP */ enum gprs_llc_primitive { /* GMM <-> LLME */ LLGMM_ASSIGN_REQ, /* GMM tells us new TLLI: TLLI old, TLLI new, Kc, CiphAlg */ LLGMM_RESET_REQ, /* GMM tells us to perform XID negotiation: TLLI */ LLGMM_RESET_CNF, /* LLC informs GMM that XID has completed: TLLI */ LLGMM_SUSPEND_REQ, /* GMM tells us MS has suspended: TLLI, Page */ LLGMM_RESUME_REQ, /* GMM tells us MS has resumed: TLLI */ LLGMM_PAGE_IND, /* LLC asks GMM to page MS: TLLI */ LLGMM_IOV_REQ, /* GMM tells us to perform XID: TLLI */ LLGMM_STATUS_IND, /* LLC informs GMM about error: TLLI, Cause */ /* LLE <-> (GMM/SNDCP/SMS/TOM) */ LL_RESET_IND, /* TLLI */ LL_ESTABLISH_REQ, /* TLLI, XID Req */ LL_ESTABLISH_IND, /* TLLI, XID Req, N201-I, N201-U */ LL_ESTABLISH_RESP, /* TLLI, XID Negotiated */ LL_ESTABLISH_CONF, /* TLLI, XID Neg, N201-i, N201-U */ LL_RELEASE_REQ, /* TLLI, Local */ LL_RELEASE_IND, /* TLLI, Cause */ LL_RELEASE_CONF, /* TLLI */ LL_XID_REQ, /* TLLI, XID Requested */ LL_XID_IND, /* TLLI, XID Req, N201-I, N201-U */ LL_XID_RESP, /* TLLI, XID Negotiated */ LL_XID_CONF, /* TLLI, XID Neg, N201-I, N201-U */ LL_DATA_REQ, /* TLLI, SN-PDU, Ref, QoS, Radio Prio, Ciph */ LL_DATA_IND, /* TLLI, SN-PDU */ LL_DATA_CONF, /* TLLI, Ref */ LL_UNITDATA_REQ, /* TLLI, SN-PDU, Ref, QoS, Radio Prio, Ciph */ LL_UNITDATA_IND, /* TLLI, SN-PDU */ LL_STATUS_IND, /* TLLI, Cause */ }; /* Section 4.5.2 Logical Link States + Annex C.2 */ enum gprs_llc_lle_state { GPRS_LLES_UNASSIGNED = 1, /* No TLLI yet */ GPRS_LLES_ASSIGNED_ADM = 2, /* TLLI assigned */ GPRS_LLES_LOCAL_EST = 3, /* Local Establishment */ GPRS_LLES_REMOTE_EST = 4, /* Remote Establishment */ GPRS_LLES_ABM = 5, GPRS_LLES_LOCAL_REL = 6, /* Local Release */ GPRS_LLES_TIMER_REC = 7, /* Timer Recovery */ }; enum gprs_llc_llme_state { GPRS_LLMS_UNASSIGNED = 1, /* No TLLI yet */ GPRS_LLMS_ASSIGNED = 2, /* TLLI assigned */ }; /* Section 8.9.9 LLC layer parameter default values */ struct gprs_llc_params { uint16_t iov_i_exp; uint16_t t200_201; uint16_t n200; uint16_t n201_u; uint16_t n201_i; uint16_t mD; uint16_t mU; uint16_t kD; uint16_t kU; }; /* Section 4.7.1: Logical Link Entity: One per DLCI (TLLI + SAPI) */ struct gprs_llc_lle { struct llist_head list; uint32_t sapi; struct gprs_llc_llme *llme; enum gprs_llc_lle_state state; struct timer_list t200; struct timer_list t201; /* wait for acknowledgement */ uint16_t v_sent; uint16_t v_ack; uint16_t v_recv; uint16_t vu_send; uint16_t vu_recv; /* Overflow Counter for ABM */ uint32_t oc_i_send; uint32_t oc_i_recv; /* Overflow Counter for unconfirmed transfer */ uint32_t oc_ui_send; uint32_t oc_ui_recv; unsigned int retrans_ctr; struct gprs_llc_params params; }; #define NUM_SAPIS 16 struct gprs_llc_llme { struct llist_head list; enum gprs_llc_llme_state state; uint32_t tlli; uint32_t old_tlli; /* Crypto parameters */ enum gprs_ciph_algo algo; uint8_t kc[8]; /* over which BSSGP BTS ctx do we need to transmit */ uint16_t bvci; uint16_t nsei; struct gprs_llc_lle lle[NUM_SAPIS]; }; extern struct llist_head gprs_llc_llmes; /* BSSGP-UL-UNITDATA.ind */ int gprs_llc_rcvmsg(struct msgb *msg, struct tlv_parsed *tv); /* LL-UNITDATA.req */ int gprs_llc_tx_ui(struct msgb *msg, uint8_t sapi, int command, void *mmctx); /* 04.64 Chapter 7.2.1.1 LLGMM-ASSIGN */ int gprs_llgmm_assign(struct gprs_llc_llme *llme, uint32_t old_tlli, uint32_t new_tlli, enum gprs_ciph_algo alg, const uint8_t *kc); int gprs_llc_init(const char *cipher_plugin_path); int gprs_llc_vty_init(void); #endif