ecl/src/c/num_arith.d
Daniel Kochmański 4d19a27424 cosmetic: untabify
Signed-off-by: Daniel Kochmański <dkochmanski@turtle-solutions.eu>
2015-06-21 14:38:20 +02:00

111 lines
3.4 KiB
C

/* -*- mode: c; c-basic-offset: 8 -*- */
/*
num_arith.c -- Arithmetic operations
*/
/*
Copyright (c) 1990, Giuseppe Attardi.
Copyright (c) 2001, Juan Jose Garcia Ripoll.
ECL is free software; you can redistribute it and/or
modify it under the terms of the GNU Library General Public
License as published by the Free Software Foundation; either
version 2 of the License, or (at your option) any later version.
See file '../Copyright' for full details.
*/
#include <ecl/ecl.h>
#include <ecl/number.h>
#include <stdlib.h>
cl_object
ecl_integer_divide(cl_object x, cl_object y)
{
cl_type tx, ty;
tx = ecl_t_of(x);
ty = ecl_t_of(y);
if (tx == t_fixnum) {
if (ty == t_fixnum) {
if (y == ecl_make_fixnum(0))
FEdivision_by_zero(x, y);
return ecl_make_fixnum(ecl_fixnum(x) / ecl_fixnum(y));
} else if (ty == t_bignum) {
return _ecl_fix_divided_by_big(ecl_fixnum(x), y);
} else {
FEwrong_type_nth_arg(@[round], 2, y, @[integer]);
}
}
if (tx == t_bignum) {
if (ty == t_bignum) {
return _ecl_big_divided_by_big(x, y);
} else if (ty == t_fixnum) {
return _ecl_big_divided_by_fix(x, ecl_fixnum(y));
} else {
FEwrong_type_nth_arg(@[round], 2, y, @[integer]);
}
}
FEwrong_type_nth_arg(@[round], 1, x, @[integer]);
}
@(defun gcd (&rest nums)
cl_object gcd;
@
if (narg == 0)
@(return ecl_make_fixnum(0))
/* INV: ecl_gcd() checks types */
gcd = ecl_va_arg(nums);
if (narg == 1) {
assert_type_integer(gcd);
@(return (ecl_minusp(gcd) ? ecl_negate(gcd) : gcd))
}
while (--narg)
gcd = ecl_gcd(gcd, ecl_va_arg(nums));
@(return gcd)
@)
cl_object
ecl_gcd(cl_object x, cl_object y)
{
ECL_WITH_TEMP_BIGNUM(x_big,1);
ECL_WITH_TEMP_BIGNUM(y_big,1);
switch (ecl_t_of(x)) {
case t_fixnum:
_ecl_big_set_fixnum(x_big, ecl_fixnum(x));
x = x_big;
case t_bignum:
break;
default:
FEwrong_type_nth_arg(@[gcd], 1, x, @[integer]);
}
switch (ecl_t_of(y)) {
case t_fixnum:
_ecl_big_set_fixnum(y_big, ecl_fixnum(y));
y = y_big;
case t_bignum:
break;
default:
FEwrong_type_nth_arg(@[gcd], 2, y, @[integer]);
}
return _ecl_big_gcd(x, y);
}
@(defun lcm (&rest nums)
cl_object lcm;
@
if (narg == 0)
@(return ecl_make_fixnum(1))
/* INV: ecl_gcd() checks types. By placing `numi' before `lcm' in
this call, we make sure that errors point to `numi' */
lcm = ecl_va_arg(nums);
assert_type_integer(lcm);
while (narg-- > 1) {
cl_object numi = ecl_va_arg(nums);
cl_object t = ecl_times(lcm, numi);
cl_object g = ecl_gcd(numi, lcm);
if (g != ecl_make_fixnum(0))
lcm = ecl_divide(t, g);
}
@(return (ecl_minusp(lcm) ? ecl_negate(lcm) : lcm))
@)