arb/acb_modular/addseq.c
Tommy Hofmann 6bf072eb59 Replace abort with flint_abort.
This will allow us to not loose the julia session on error.
See also https://github.com/wbhart/flint2/pull/243
2017-02-28 16:52:57 +01:00

178 lines
3.4 KiB
C

/*
Copyright (C) 2014 Fredrik Johansson
This file is part of Arb.
Arb is free software: you can redistribute it and/or modify it under
the terms of the GNU Lesser General Public License (LGPL) as published
by the Free Software Foundation; either version 2.1 of the License, or
(at your option) any later version. See <http://www.gnu.org/licenses/>.
*/
#include "acb_modular.h"
static slong
bisect(slong needle, const slong * haystack, slong len)
{
slong a, b, mid;
a = 0;
b = len - 1;
while (a < b)
{
mid = (a + b) / 2;
if (haystack[mid] < needle)
a = mid + 1;
else
b = mid;
}
if (a == b && haystack[a] == needle)
{
return a;
}
return -1;
}
/* write p = 2a with a in P */
static int
write_as_2a(slong * i1, slong * i2, slong p, const slong * P, slong Plen)
{
if (p % 2 == 0)
{
slong i = bisect(p / 2, P, Plen);
if (i != -1)
{
*i1 = *i2 = i;
return 1;
}
}
return 0;
}
/* write p = a + b with a, b in P */
static int
write_as_a_b(slong * i1, slong * i2, slong p, const slong * P, slong Plen)
{
slong i, j, pi;
for (i = 0; i < Plen; i++)
{
pi = P[i];
j = bisect(p - pi, P, Plen);
if (j != -1)
{
*i1 = i;
*i2 = j;
return 1;
}
}
return 0;
}
/* write p = 2a + b with a, b in P */
static int
write_as_2a_b(slong * i1, slong * i2, slong p, const slong * P, slong Plen)
{
slong i, j, pi;
for (i = 0; i < Plen; i++)
{
pi = P[i];
if (2 * pi > p)
break;
j = bisect(p - 2*pi, P, Plen);
if (j != -1)
{
*i1 = i;
*i2 = j;
return 1;
}
}
return 0;
}
void
acb_modular_addseq_theta(slong * exponents, slong * aindex, slong * bindex, slong num)
{
slong i;
slong c;
for (i = 0; i < num; i++)
{
if (i == 0)
{
exponents[i] = 1;
aindex[i] = 0;
bindex[i] = 0;
continue;
}
else
{
c = ((i + 2) * (i + 2)) / 4;
exponents[i] = c;
if (write_as_2a(aindex + i, bindex + i, c, exponents, i))
continue;
if (write_as_a_b(aindex + i, bindex + i, c, exponents, i))
continue;
if (write_as_2a_b(aindex + i, bindex + i, c, exponents, i))
continue;
flint_printf("i = %wd, c = %wu: bad addition sequence!\n", i, c);
flint_abort();
}
}
}
void
acb_modular_addseq_eta(slong * exponents, slong * aindex, slong * bindex, slong num)
{
slong i;
slong c;
for (i = 0; i < num; i++)
{
if (i == 0)
{
exponents[i] = 1;
aindex[i] = 0;
bindex[i] = 0;
continue;
}
else
{
c = ((i + 2) / 2) * ((3 * i + 5) / 2) / 2;
exponents[i] = c;
if (write_as_2a(aindex + i, bindex + i, c, exponents, i))
continue;
if (write_as_a_b(aindex + i, bindex + i, c, exponents, i))
continue;
if (write_as_2a_b(aindex + i, bindex + i, c, exponents, i))
continue;
flint_printf("i = %wd, c = %wu: bad addition sequence!\n", i, c);
flint_abort();
}
}
}