/*
 * This file is part of the TREZOR project, https://trezor.io/
 *
 * Copyright (c) SatoshiLabs
 *
 * This program 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.
 *
 * 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.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 */

#include "py/objstr.h"

#include "groestl.h"
#include "memzero.h"

#define GROESTL512_DIGEST_LENGTH 64
#define GROESTL512_BLOCK_LENGTH 128

/// package: trezorcrypto.__init__

/// class Groestl512:
///     '''
///     GROESTL512 context.
///     '''
typedef struct _mp_obj_Groestl512_t {
  mp_obj_base_t base;
  GROESTL512_CTX ctx;
} mp_obj_Groestl512_t;

STATIC mp_obj_t mod_trezorcrypto_Groestl512_update(mp_obj_t self,
                                                   mp_obj_t data);

/// def __init__(self, data: bytes = None) -> None:
///     '''
///     Creates a hash context object.
///     '''
STATIC mp_obj_t mod_trezorcrypto_Groestl512_make_new(const mp_obj_type_t *type,
                                                     size_t n_args, size_t n_kw,
                                                     const mp_obj_t *args) {
  mp_arg_check_num(n_args, n_kw, 0, 1, false);
  mp_obj_Groestl512_t *o = m_new_obj(mp_obj_Groestl512_t);
  o->base.type = type;
  groestl512_Init(&(o->ctx));
  if (n_args == 1) {
    mod_trezorcrypto_Groestl512_update(MP_OBJ_FROM_PTR(o), args[0]);
  }
  return MP_OBJ_FROM_PTR(o);
}

/// def update(self, data: bytes) -> None:
///     '''
///     Update the hash context with hashed data.
///     '''
STATIC mp_obj_t mod_trezorcrypto_Groestl512_update(mp_obj_t self,
                                                   mp_obj_t data) {
  mp_obj_Groestl512_t *o = MP_OBJ_TO_PTR(self);
  mp_buffer_info_t msg;
  mp_get_buffer_raise(data, &msg, MP_BUFFER_READ);
  if (msg.len > 0) {
    groestl512_Update(&(o->ctx), msg.buf, msg.len);
  }
  return mp_const_none;
}
STATIC MP_DEFINE_CONST_FUN_OBJ_2(mod_trezorcrypto_Groestl512_update_obj,
                                 mod_trezorcrypto_Groestl512_update);

/// def digest(self) -> bytes:
///     '''
///     Returns the digest of hashed data.
///     '''
STATIC mp_obj_t mod_trezorcrypto_Groestl512_digest(mp_obj_t self) {
  mp_obj_Groestl512_t *o = MP_OBJ_TO_PTR(self);
  uint8_t out[GROESTL512_DIGEST_LENGTH];
  GROESTL512_CTX ctx;
  memcpy(&ctx, &(o->ctx), sizeof(GROESTL512_CTX));
  groestl512_Final(&ctx, out);
  memzero(&ctx, sizeof(GROESTL512_CTX));
  return mp_obj_new_bytes(out, sizeof(out));
}
STATIC MP_DEFINE_CONST_FUN_OBJ_1(mod_trezorcrypto_Groestl512_digest_obj,
                                 mod_trezorcrypto_Groestl512_digest);

STATIC mp_obj_t mod_trezorcrypto_Groestl512___del__(mp_obj_t self) {
  mp_obj_Groestl512_t *o = MP_OBJ_TO_PTR(self);
  memzero(&(o->ctx), sizeof(GROESTL512_CTX));
  return mp_const_none;
}
STATIC MP_DEFINE_CONST_FUN_OBJ_1(mod_trezorcrypto_Groestl512___del___obj,
                                 mod_trezorcrypto_Groestl512___del__);

STATIC const mp_rom_map_elem_t
    mod_trezorcrypto_Groestl512_locals_dict_table[] = {
        {MP_ROM_QSTR(MP_QSTR_update),
         MP_ROM_PTR(&mod_trezorcrypto_Groestl512_update_obj)},
        {MP_ROM_QSTR(MP_QSTR_digest),
         MP_ROM_PTR(&mod_trezorcrypto_Groestl512_digest_obj)},
        {MP_ROM_QSTR(MP_QSTR___del__),
         MP_ROM_PTR(&mod_trezorcrypto_Groestl512___del___obj)},
        {MP_ROM_QSTR(MP_QSTR_block_size),
         MP_OBJ_NEW_SMALL_INT(GROESTL512_BLOCK_LENGTH)},
        {MP_ROM_QSTR(MP_QSTR_digest_size),
         MP_OBJ_NEW_SMALL_INT(GROESTL512_DIGEST_LENGTH)},
};
STATIC MP_DEFINE_CONST_DICT(mod_trezorcrypto_Groestl512_locals_dict,
                            mod_trezorcrypto_Groestl512_locals_dict_table);

STATIC const mp_obj_type_t mod_trezorcrypto_Groestl512_type = {
    {&mp_type_type},
    .name = MP_QSTR_Groestl512,
    .make_new = mod_trezorcrypto_Groestl512_make_new,
    .locals_dict = (void *)&mod_trezorcrypto_Groestl512_locals_dict,
};