2021-03-23 12:14:33 +00:00
|
|
|
/*
|
|
|
|
* 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 <string.h>
|
|
|
|
|
|
|
|
#include "memzero.h"
|
2021-09-09 14:53:05 +00:00
|
|
|
#include "py/obj.h"
|
2021-03-23 12:14:33 +00:00
|
|
|
#include "py/objint.h"
|
2021-09-09 14:53:05 +00:00
|
|
|
#include "py/objstr.h"
|
2021-06-22 17:47:03 +00:00
|
|
|
#include "py/runtime.h"
|
2021-03-23 12:14:33 +00:00
|
|
|
|
|
|
|
static bool mpz_as_ll_checked(const mpz_t *i, long long *value) {
|
|
|
|
// Analogue of `mpz_as_int_checked` from mpz.c
|
|
|
|
|
|
|
|
unsigned long long val = 0;
|
|
|
|
mpz_dig_t *d = i->dig + i->len;
|
|
|
|
|
|
|
|
while (d-- > i->dig) {
|
|
|
|
if (val > (~0x8000000000000000 >> MPZ_DIG_SIZE)) {
|
|
|
|
// will overflow
|
|
|
|
*value = 0;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
val = (val << MPZ_DIG_SIZE) | *d;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i->neg != 0) {
|
|
|
|
val = -val;
|
|
|
|
}
|
|
|
|
|
|
|
|
*value = val;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool trezor_obj_get_ll_checked(mp_obj_t obj, long long *value) {
|
|
|
|
if (mp_obj_is_small_int(obj)) {
|
|
|
|
// Value is fitting in a small int range. Return it directly.
|
|
|
|
*value = MP_OBJ_SMALL_INT_VALUE(obj);
|
|
|
|
return true;
|
|
|
|
|
2023-01-24 20:57:27 +00:00
|
|
|
} else if (mp_obj_is_int(obj)) {
|
2021-03-23 12:14:33 +00:00
|
|
|
// Value is not fitting into small int range, but is an integer.
|
|
|
|
mp_obj_int_t *self = MP_OBJ_TO_PTR(obj);
|
|
|
|
// Try to get the long long value out of the MPZ struct.
|
|
|
|
return mpz_as_ll_checked(&self->mpz, value);
|
|
|
|
} else {
|
|
|
|
// Value is not integer.
|
|
|
|
*value = 0;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2021-06-22 17:47:03 +00:00
|
|
|
|
|
|
|
mp_obj_t trezor_obj_call_protected(void (*func)(void *), void *arg) {
|
|
|
|
nlr_buf_t nlr;
|
|
|
|
if (nlr_push(&nlr) == 0) {
|
|
|
|
(*func)(arg);
|
|
|
|
nlr_pop();
|
2021-09-16 08:55:36 +00:00
|
|
|
return NULL;
|
2021-06-22 17:47:03 +00:00
|
|
|
} else {
|
|
|
|
return MP_OBJ_FROM_PTR(nlr.ret_val);
|
|
|
|
}
|
|
|
|
}
|
2021-09-09 14:53:05 +00:00
|
|
|
|
|
|
|
mp_obj_t trezor_obj_str_from_rom_text(const char *str) {
|
|
|
|
// taken from mp_obj_new_exception_msg
|
|
|
|
mp_obj_str_t *o_str = m_new_obj_maybe(mp_obj_str_t);
|
|
|
|
if (o_str == NULL) return NULL;
|
|
|
|
|
|
|
|
o_str->base.type = &mp_type_str;
|
|
|
|
o_str->len = strlen(str);
|
|
|
|
o_str->data = (const byte *)str;
|
|
|
|
#if MICROPY_ROM_TEXT_COMPRESSION
|
|
|
|
o_str->hash = 0; // will be computed only if string object is accessed
|
|
|
|
#else
|
|
|
|
o_str->hash = qstr_compute_hash(o_str->data, o_str->len);
|
|
|
|
#endif
|
|
|
|
return MP_OBJ_FROM_PTR(o_str);
|
|
|
|
}
|