|
|
|
/*
|
|
|
|
* Copyright 2013-2019 Software Radio Systems Limited
|
|
|
|
*
|
|
|
|
* This file is part of srsLTE.
|
|
|
|
*
|
|
|
|
* srsLTE is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero General Public License as
|
|
|
|
* published by the Free Software Foundation, either version 3 of
|
|
|
|
* the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* srsLTE is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU Affero General Public License for more details.
|
|
|
|
*
|
|
|
|
* A copy of the GNU Affero General Public License can be found in
|
|
|
|
* the LICENSE file in the top-level directory of this distribution
|
|
|
|
* and at http://www.gnu.org/licenses/.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "srslte/upper/pdcp_entity_base.h"
|
|
|
|
#include "srslte/common/security.h"
|
|
|
|
|
|
|
|
namespace srslte {
|
|
|
|
|
|
|
|
pdcp_entity_base::pdcp_entity_base() {}
|
|
|
|
|
|
|
|
pdcp_entity_base::~pdcp_entity_base() {}
|
|
|
|
|
|
|
|
void pdcp_entity_base::config_security(uint8_t* k_rrc_enc_,
|
|
|
|
uint8_t* k_rrc_int_,
|
|
|
|
uint8_t* k_up_enc_,
|
|
|
|
uint8_t* k_up_int_,
|
|
|
|
CIPHERING_ALGORITHM_ID_ENUM cipher_algo_,
|
|
|
|
INTEGRITY_ALGORITHM_ID_ENUM integ_algo_)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < 32; i++) {
|
|
|
|
k_rrc_enc[i] = k_rrc_enc_[i];
|
|
|
|
k_rrc_int[i] = k_rrc_int_[i];
|
|
|
|
k_up_enc[i] = k_up_enc_[i];
|
|
|
|
if (k_up_int_ != nullptr) {
|
|
|
|
k_up_int[i] = k_up_int_[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
cipher_algo = cipher_algo_;
|
|
|
|
integ_algo = integ_algo_;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Security functions
|
|
|
|
***************************************************************************/
|
|
|
|
void pdcp_entity_base::integrity_generate(uint8_t* msg, uint32_t msg_len, uint32_t count, uint8_t* mac)
|
|
|
|
{
|
|
|
|
uint8_t *k_int;
|
|
|
|
|
|
|
|
// If control plane use RRC integrity key. If data use user plane key
|
|
|
|
if (is_srb()) {
|
|
|
|
k_int = k_rrc_int;
|
|
|
|
} else {
|
|
|
|
k_int = k_up_int;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(integ_algo)
|
|
|
|
{
|
|
|
|
case INTEGRITY_ALGORITHM_ID_EIA0:
|
|
|
|
break;
|
|
|
|
case INTEGRITY_ALGORITHM_ID_128_EIA1:
|
|
|
|
security_128_eia1(&k_int[16],
|
|
|
|
count,
|
|
|
|
cfg.bearer_id - 1,
|
|
|
|
cfg.direction,
|
|
|
|
msg,
|
|
|
|
msg_len,
|
|
|
|
mac);
|
|
|
|
break;
|
|
|
|
case INTEGRITY_ALGORITHM_ID_128_EIA2:
|
|
|
|
security_128_eia2(&k_int[16],
|
|
|
|
count,
|
|
|
|
cfg.bearer_id - 1,
|
|
|
|
cfg.direction,
|
|
|
|
msg,
|
|
|
|
msg_len,
|
|
|
|
mac);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
log->debug("Integrity gen input: COUNT %d, Bearer ID %d, Direction %s\n",
|
|
|
|
count,
|
|
|
|
cfg.bearer_id,
|
|
|
|
(cfg.direction == SECURITY_DIRECTION_DOWNLINK ? "Downlink" : "Uplink"));
|
|
|
|
log->debug_hex(mac, 4, "MAC (generated)");
|
|
|
|
log->debug_hex(msg, msg_len, " Message");
|
|
|
|
}
|
|
|
|
|
|
|
|
bool pdcp_entity_base::integrity_verify(uint8_t* msg, uint32_t msg_len, uint32_t count, uint8_t* mac)
|
|
|
|
{
|
|
|
|
uint8_t mac_exp[4] = {};
|
|
|
|
bool is_valid = true;
|
|
|
|
uint8_t *k_int;
|
|
|
|
|
|
|
|
// If control plane use RRC integrity key. If data use user plane key
|
|
|
|
if (is_srb()) {
|
|
|
|
k_int = k_rrc_int;
|
|
|
|
} else {
|
|
|
|
k_int = k_up_int;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (integ_algo) {
|
|
|
|
case INTEGRITY_ALGORITHM_ID_EIA0:
|
|
|
|
break;
|
|
|
|
case INTEGRITY_ALGORITHM_ID_128_EIA1:
|
|
|
|
security_128_eia1(&k_int[16],
|
|
|
|
count,
|
|
|
|
cfg.bearer_id - 1,
|
|
|
|
(cfg.direction == SECURITY_DIRECTION_DOWNLINK) ? (SECURITY_DIRECTION_UPLINK)
|
|
|
|
: (SECURITY_DIRECTION_DOWNLINK),
|
|
|
|
msg,
|
|
|
|
msg_len,
|
|
|
|
mac_exp);
|
|
|
|
break;
|
|
|
|
case INTEGRITY_ALGORITHM_ID_128_EIA2:
|
|
|
|
security_128_eia2(&k_int[16],
|
|
|
|
count,
|
|
|
|
cfg.bearer_id - 1,
|
|
|
|
(cfg.direction == SECURITY_DIRECTION_DOWNLINK) ? (SECURITY_DIRECTION_UPLINK)
|
|
|
|
: (SECURITY_DIRECTION_DOWNLINK),
|
|
|
|
msg,
|
|
|
|
msg_len,
|
|
|
|
mac_exp);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
log->debug("Integrity check input: COUNT %d, Bearer ID %d, Direction %s\n",
|
|
|
|
count,
|
|
|
|
cfg.bearer_id,
|
|
|
|
(cfg.direction == SECURITY_DIRECTION_DOWNLINK ? "Downlink" : "Uplink"));
|
|
|
|
log->debug_hex(msg, msg_len, " Message");
|
|
|
|
|
|
|
|
if (integ_algo != INTEGRITY_ALGORITHM_ID_EIA0) {
|
|
|
|
for (uint8_t i = 0; i < 4; i++) {
|
|
|
|
if (mac[i] != mac_exp[i]) {
|
|
|
|
log->error_hex(mac_exp, 4, "MAC mismatch (expected)");
|
|
|
|
log->error_hex(mac, 4, "MAC mismatch (found)");
|
|
|
|
is_valid = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (is_valid) {
|
|
|
|
log->info_hex(mac_exp, 4, "MAC match");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return is_valid;
|
|
|
|
}
|
|
|
|
|
|
|
|
void pdcp_entity_base::cipher_encrypt(uint8_t* msg, uint32_t msg_len, uint32_t count, uint8_t* ct)
|
|
|
|
{
|
|
|
|
byte_buffer_t ct_tmp;
|
|
|
|
uint8_t *k_enc;
|
|
|
|
|
|
|
|
// If control plane use RRC encrytion key. If data use user plane key
|
|
|
|
if (is_srb()) {
|
|
|
|
k_enc = k_rrc_enc;
|
|
|
|
} else {
|
|
|
|
k_enc = k_up_enc;
|
|
|
|
}
|
|
|
|
|
|
|
|
log->debug("Cipher encrypt input: COUNT: %d, Bearer ID: %d, Direction %s\n",
|
|
|
|
count,
|
|
|
|
cfg.bearer_id,
|
|
|
|
(cfg.direction == SECURITY_DIRECTION_DOWNLINK) ? "Downlink" : "Uplink");
|
|
|
|
|
|
|
|
switch (cipher_algo) {
|
|
|
|
case CIPHERING_ALGORITHM_ID_EEA0:
|
|
|
|
break;
|
|
|
|
case CIPHERING_ALGORITHM_ID_128_EEA1:
|
|
|
|
security_128_eea1(&(k_enc[16]), count, cfg.bearer_id - 1, cfg.direction, msg, msg_len, ct_tmp.msg);
|
|
|
|
memcpy(ct, ct_tmp.msg, msg_len);
|
|
|
|
break;
|
|
|
|
case CIPHERING_ALGORITHM_ID_128_EEA2:
|
|
|
|
security_128_eea2(&(k_enc[16]), count, cfg.bearer_id - 1, cfg.direction, msg, msg_len, ct_tmp.msg);
|
|
|
|
memcpy(ct, ct_tmp.msg, msg_len);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void pdcp_entity_base::cipher_decrypt(uint8_t* ct, uint32_t ct_len, uint32_t count, uint8_t* msg)
|
|
|
|
{
|
|
|
|
byte_buffer_t msg_tmp;
|
|
|
|
uint8_t *k_enc;
|
|
|
|
// If control plane use RRC encrytion key. If data use user plane key
|
|
|
|
if (is_srb()) {
|
|
|
|
k_enc = k_rrc_enc;
|
|
|
|
} else {
|
|
|
|
k_enc = k_up_enc;
|
|
|
|
}
|
|
|
|
|
|
|
|
log->debug("Cipher decript input: COUNT: %d, Bearer ID: %d, Direction %s\n",
|
|
|
|
count,
|
|
|
|
cfg.bearer_id,
|
|
|
|
(cfg.direction == SECURITY_DIRECTION_DOWNLINK) ? "Downlink" : "Uplink");
|
|
|
|
|
|
|
|
switch(cipher_algo)
|
|
|
|
{
|
|
|
|
case CIPHERING_ALGORITHM_ID_EEA0:
|
|
|
|
break;
|
|
|
|
case CIPHERING_ALGORITHM_ID_128_EEA1:
|
|
|
|
security_128_eea1(&(k_enc[16]),
|
|
|
|
count,
|
|
|
|
cfg.bearer_id - 1,
|
|
|
|
(cfg.direction == SECURITY_DIRECTION_DOWNLINK) ? (SECURITY_DIRECTION_UPLINK)
|
|
|
|
: (SECURITY_DIRECTION_DOWNLINK),
|
|
|
|
ct,
|
|
|
|
ct_len,
|
|
|
|
msg_tmp.msg);
|
|
|
|
memcpy(msg, msg_tmp.msg, ct_len);
|
|
|
|
break;
|
|
|
|
case CIPHERING_ALGORITHM_ID_128_EEA2:
|
|
|
|
security_128_eea2(&(k_enc[16]),
|
|
|
|
count,
|
|
|
|
cfg.bearer_id - 1,
|
|
|
|
(cfg.direction == SECURITY_DIRECTION_DOWNLINK) ? (SECURITY_DIRECTION_UPLINK)
|
|
|
|
: (SECURITY_DIRECTION_DOWNLINK),
|
|
|
|
ct,
|
|
|
|
ct_len,
|
|
|
|
msg_tmp.msg);
|
|
|
|
memcpy(msg, msg_tmp.msg, ct_len);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|