Skip to content
Snippets Groups Projects
nas_stream_eea2.c 2.97 KiB
Newer Older
gauthier's avatar
gauthier committed
/*******************************************************************************
    OpenAirInterface
    Copyright(c) 1999 - 2014 Eurecom

    OpenAirInterface is free software: you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
    the Free Software Foundation, either version 3 of the License, or
    (at your option) any later version.


    OpenAirInterface 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 General Public License for more details.

    You should have received a copy of the GNU General Public License
    along with OpenAirInterface.The full GNU General Public License is
   included in this distribution in the file called "COPYING". If not,
   see <http://www.gnu.org/licenses/>.

  Contact Information
  OpenAirInterface Admin: openair_admin@eurecom.fr
  OpenAirInterface Tech : openair_tech@eurecom.fr
  OpenAirInterface Dev  : openair4g-devel@lists.eurecom.fr
gauthier's avatar
gauthier committed

  Address      : Eurecom, Compus SophiaTech 450, route des chappes, 06451 Biot, France.

 *******************************************************************************/
Cédric Roux's avatar
Cédric Roux committed
#include <stdlib.h>
#include <stdio.h>
#include <stdint.h>
#include <string.h>

#include <nettle/nettle-meta.h>
#include <nettle/aes.h>
#include <nettle/ctr.h>

#include "assertions.h"
#include "conversions.h"
#include "secu_defs.h"

// #define SECU_DEBUG

int nas_stream_encrypt_eea2(nas_stream_cipher_t *stream_cipher, uint8_t *out)
Cédric Roux's avatar
Cédric Roux committed
{
  uint8_t m[16];
  uint32_t local_count;
Cédric Roux's avatar
Cédric Roux committed

Cédric Roux's avatar
Cédric Roux committed

  uint32_t zero_bit = 0;
  uint32_t byte_length;
Cédric Roux's avatar
Cédric Roux committed

  DevAssert(stream_cipher != NULL);
  DevAssert(out != NULL);
Cédric Roux's avatar
Cédric Roux committed

  zero_bit = stream_cipher->blength & 0x7;
Cédric Roux's avatar
Cédric Roux committed

  byte_length = stream_cipher->blength >> 3;
Cédric Roux's avatar
Cédric Roux committed

Cédric Roux's avatar
Cédric Roux committed

  ctx = malloc(nettle_aes128.context_size);
  data = malloc(byte_length);
Cédric Roux's avatar
Cédric Roux committed

  local_count = hton_int32(stream_cipher->count);

  memset(m, 0, sizeof(m));
  memcpy(&m[0], &local_count, 4);
  m[4] = ((stream_cipher->bearer & 0x1F) << 3) |
         ((stream_cipher->direction & 0x01) << 2);
  /* Other bits are 0 */
Cédric Roux's avatar
Cédric Roux committed

#if defined(SECU_DEBUG)
  {
    int i;
    printf("Blength: %u, Zero bits: %u\n", stream_cipher->blength, zero_bit);

    for (i = 0; i < sizeof(m); i++)
      printf("0x%02x ", m[i]);

    printf("\n");
  }
Cédric Roux's avatar
Cédric Roux committed
#endif

#if NETTLE_VERSION <= 27
  nettle_aes128.set_encrypt_key(ctx, stream_cipher->key_length,
                                stream_cipher->key);
#else
Cédric Roux's avatar
Cédric Roux committed

  nettle_ctr_crypt(ctx, nettle_aes128.encrypt,
                   nettle_aes128.block_size, m,
                   byte_length, data, stream_cipher->message);
Cédric Roux's avatar
Cédric Roux committed

  if (zero_bit > 0)
    data[byte_length - 1] = data[byte_length - 1] & (uint8_t)(0xFF << (8 - zero_bit));
Cédric Roux's avatar
Cédric Roux committed


  memcpy(out, data, byte_length);
  free(data);
  free(ctx);
Cédric Roux's avatar
Cédric Roux committed

Cédric Roux's avatar
Cédric Roux committed
}