summaryrefslogtreecommitdiff
path: root/ext/rational/rational.c
blob: 18024e9497d16f133d4f837cc992987b47e83633 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
#include "ruby.h"

/*
 * call-seq:
 *    fixnum.gcd(fixnum)  ->  fixnum
 *
 * Fixnum-specific optimized version of Integer#gcd.  Delegates to
 * Integer#gcd as necessary.
 */
static VALUE
fix_gcd(self, other)
    VALUE self, other;
{
    long a, b, min, max;

    /*
     * Note: Cannot handle values <= FIXNUM_MIN here due to overflow during negation.
     */
    if (!FIXNUM_P(other) ||
	(a = FIX2LONG(self)) <= FIXNUM_MIN ||
	(b = FIX2LONG(other)) <= FIXNUM_MIN ) {
	/* Delegate to Integer#gcd */
	return rb_call_super(1, &other);
    }

    min = a < 0 ? -a : a;
    max = b < 0 ? -b : b;

    while (min > 0) {
	long tmp = min;
	min = max % min;
	max = tmp;
    }

    return LONG2FIX(max);
}

void
Init_rational()
{
    rb_define_method(rb_cFixnum, "gcd", fix_gcd, 1);
}