aboutsummaryrefslogtreecommitdiff
path: root/builtins
diff options
context:
space:
mode:
authorBruce Hill <bruce@bruce-hill.com>2024-08-13 16:32:00 -0400
committerBruce Hill <bruce@bruce-hill.com>2024-08-13 16:32:00 -0400
commite0223a4c201b42ca35c73bfad84477efc61f06b2 (patch)
tree7f83903c19433c67e102a8688c3af02fd63eab0a /builtins
parentaaac5dff28b2fa4835dd9a3a352f274069a7b432 (diff)
Support ^ exponentiation for integers
Diffstat (limited to 'builtins')
-rw-r--r--builtins/integers.c11
-rw-r--r--builtins/integers.h7
2 files changed, 16 insertions, 2 deletions
diff --git a/builtins/integers.c b/builtins/integers.c
index 53e38aab..e55d2ab0 100644
--- a/builtins/integers.c
+++ b/builtins/integers.c
@@ -291,6 +291,17 @@ public Int_t Int$abs(Int_t x)
return Int$from_mpz(result);
}
+public Int_t Int$power(Int_t base, Int_t exponent)
+{
+ int64_t exp = Int$as_i64(exponent);
+ if (__builtin_expect(exp < 0, 0))
+ fail("Cannot take a negative power of an integer!");
+ mpz_t result;
+ mpz_init_set_int(result, base);
+ mpz_pow_ui(result, result, exp);
+ return Int$from_mpz(result);
+}
+
public Int_t Int$random(Int_t min, Int_t max) {
int32_t cmp = Int$compare(&min, &max, &$Int);
if (cmp > 0)
diff --git a/builtins/integers.h b/builtins/integers.h
index 385113b0..8250e0dc 100644
--- a/builtins/integers.h
+++ b/builtins/integers.h
@@ -59,6 +59,7 @@ Int_t Int$random(Int_t min, Int_t max);
Range_t Int$to(Int_t from, Int_t to);
Int_t Int$from_text(CORD text);
Int_t Int$abs(Int_t x);
+Int_t Int$power(Int_t base, Int_t exponent);
#define BIGGEST_SMALL_INT ((1<<29)-1)
@@ -72,11 +73,13 @@ Int_t Int$abs(Int_t x);
}))
#define mpz_init_set_int(mpz, i) do { \
- if ((i).small & 1) mpz_init_set_si(mpz, (i).small >> 2); \
+ if (__builtin_expect((i).small & 1, 1)) mpz_init_set_si(mpz, (i).small >> 2); \
else mpz_init_set(mpz, *(i).big); \
} while (0)
-#define Int$as_i64(i) (((i).small & 1) ? (int64_t)((i).small >> 2) : mpz_get_si(*(i).big))
+#define Int$as_i64(i) (__builtin_expect((i).small & 1, 1) ? (int64_t)((i).small >> 2) : \
+ ({ if (!__builtin_expect(mpz_fits_slong_p(*(i).big), 1)) fail("Integer is too big to fit in a 64-bit integer!"); \
+ mpz_get_si(*(i).big); }))
Int_t Int$from_i64(int64_t i);
Int_t Int$from_num(double n);
double Int$as_num(Int_t i);