
edu.jas.gb.GroebnerBaseSeqPairParallel Maven / Gradle / Ivy
The newest version!
/*
* $Id: GroebnerBaseSeqPairParallel.java 4059 2012-07-27 11:16:42Z kredel $
*/
package edu.jas.gb;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.ListIterator;
import java.util.concurrent.Semaphore;
import org.apache.log4j.Logger;
import edu.jas.poly.ExpVector;
import edu.jas.poly.GenPolynomial;
import edu.jas.structure.RingElem;
import edu.jas.util.Terminator;
import edu.jas.util.ThreadPool;
/**
* Groebner Base parallel algorithm. Makes some effort to produce the same
* sequence of critical pairs as in the sequential version. However already
* reduced pairs are not rereduced if new polynomials appear. Implements a
* shared memory parallel version of Groebner bases. Slaves maintain pairlist.
* @param coefficient type
* @author Heinz Kredel
*/
public class GroebnerBaseSeqPairParallel> extends GroebnerBaseAbstract {
private static final Logger logger = Logger.getLogger(GroebnerBaseSeqPairParallel.class);
/**
* Number of threads to use.
*/
protected final int threads;
/**
* Pool of threads to use.
*/
protected transient final ThreadPool pool;
/**
* Constructor.
*/
public GroebnerBaseSeqPairParallel() {
this(2);
}
/**
* Constructor.
* @param threads number of threads to use.
*/
public GroebnerBaseSeqPairParallel(int threads) {
this(threads, new ThreadPool(threads));
}
/**
* Constructor.
* @param threads number of threads to use.
* @param pool ThreadPool to use.
*/
public GroebnerBaseSeqPairParallel(int threads, ThreadPool pool) {
this(threads, pool, new ReductionPar());
}
/**
* Constructor.
* @param threads number of threads to use.
* @param red parallelism aware reduction engine
*/
public GroebnerBaseSeqPairParallel(int threads, Reduction red) {
this(threads, new ThreadPool(threads), red);
}
/**
* Constructor.
* @param threads number of threads to use.
* @param pool ThreadPool to use.
* @param red parallelism aware reduction engine
*/
public GroebnerBaseSeqPairParallel(int threads, ThreadPool pool, Reduction red) {
super(red);
if (!(red instanceof ReductionPar)) {
logger.warn("parallel GB should use parallel aware reduction");
}
if (threads < 1) {
threads = 1;
}
this.threads = threads;
this.pool = pool;
}
/**
* Cleanup and terminate ThreadPool.
*/
@Override
public void terminate() {
if (pool == null) {
return;
}
pool.terminate();
}
/**
* Cancel ThreadPool.
*/
@Override
public int cancel() {
if (pool == null) {
return 0;
}
int s = pool.cancel();
return s;
}
/**
* Parallel Groebner base using sequential pair order class. Slaves maintain
* pairlist.
* @param modv number of module variables.
* @param F polynomial list.
* @return GB(F) a Groebner base of F.
*/
public List> GB(int modv, List> F) {
GenPolynomial p;
List> G = new ArrayList>();
CriticalPairList pairlist = null;
int l = F.size();
ListIterator> it = F.listIterator();
while (it.hasNext()) {
p = it.next();
if (p.length() > 0) {
p = p.monic();
if (p.isONE()) {
G.clear();
G.add(p);
return G; // since no threads activated jet
}
G.add(p);
if (pairlist == null) {
pairlist = new CriticalPairList(modv, p.ring);
if (!p.ring.coFac.isField()) {
throw new IllegalArgumentException("coefficients not from a field");
}
}
// putOne not required
pairlist.put(p);
} else {
l--;
}
}
if (l <= 1) {
return G; // since no threads activated jet
}
Terminator fin = new Terminator(threads);
ReducerSeqPair R;
for (int i = 0; i < threads; i++) {
R = new ReducerSeqPair(fin, G, pairlist);
pool.addJob(R);
}
fin.waitDone();
if (Thread.currentThread().isInterrupted()) {
throw new RuntimeException("interrupt before minimalGB");
}
logger.debug("#parallel list = " + G.size());
G = minimalGB(G);
// not in this context // pool.terminate();
logger.info("" + pairlist);
return G;
}
/**
* Minimal ordered groebner basis, parallel.
* @param Fp a Groebner base.
* @return minimalGB(F) a minimal Groebner base of Fp.
*/
@Override
public List> minimalGB(List> Fp) {
GenPolynomial a;
ArrayList> G;
G = new ArrayList>(Fp.size());
ListIterator> it = Fp.listIterator();
while (it.hasNext()) {
a = it.next();
if (a.length() != 0) { // always true
// already monic a = a.monic();
G.add(a);
}
}
if (G.size() <= 1) {
return G;
}
ExpVector e;
ExpVector f;
GenPolynomial p;
ArrayList> F;
F = new ArrayList>(G.size());
boolean mt;
while (G.size() > 0) {
a = G.remove(0);
e = a.leadingExpVector();
it = G.listIterator();
mt = false;
while (it.hasNext() && !mt) {
p = it.next();
f = p.leadingExpVector();
mt = e.multipleOf(f);
}
it = F.listIterator();
while (it.hasNext() && !mt) {
p = it.next();
f = p.leadingExpVector();
mt = e.multipleOf(f);
}
if (!mt) {
F.add(a); // no thread at this point
} else {
// System.out.println("dropped " + a.length());
}
}
G = F;
if (G.size() <= 1) {
return G;
}
Collections.reverse(G); // important for lex GB
MiReducerSeqPair[] mirs = (MiReducerSeqPair[]) new MiReducerSeqPair[G.size()];
int i = 0;
F = new ArrayList>(G.size());
while (G.size() > 0) {
a = G.remove(0);
// System.out.println("doing " + a.length());
List> R = new ArrayList>(G.size() + F.size());
R.addAll(G);
R.addAll(F);
mirs[i] = new MiReducerSeqPair(R, a);
pool.addJob(mirs[i]);
i++;
F.add(a);
}
G = F;
F = new ArrayList>(G.size());
for (i = 0; i < mirs.length; i++) {
a = mirs[i].getNF();
F.add(a);
}
return F;
}
}
/**
* Reducing worker threads.
*/
class ReducerSeqPair> implements Runnable {
private final List> G;
private final CriticalPairList pairlist;
private final Terminator fin;
private final ReductionPar red;
private static final Logger logger = Logger.getLogger(ReducerSeqPair.class);
ReducerSeqPair(Terminator fin, List> G, CriticalPairList L) {
this.fin = fin;
this.G = G;
pairlist = L;
red = new ReductionPar();
}
/**
* to string
*/
@Override
public String toString() {
return "ReducerSeqPair";
}
public void run() {
CriticalPair pair;
GenPolynomial S;
GenPolynomial H;
boolean set = false;
int reduction = 0;
int sleeps = 0;
while (pairlist.hasNext() || fin.hasJobs()) {
while (!pairlist.hasNext()) {
pairlist.update();
// wait
fin.beIdle();
set = true;
try {
sleeps++;
if (sleeps % 10 == 0) {
logger.info(" reducer is sleeping");
} else {
logger.debug("r");
}
Thread.sleep(100);
} catch (InterruptedException e) {
break;
}
if (!fin.hasJobs()) {
break;
}
if (Thread.currentThread().isInterrupted()) {
throw new RuntimeException("interrupt after sleep");
}
}
if (!pairlist.hasNext() && !fin.hasJobs()) {
break;
}
if (set) {
fin.notIdle();
set = false;
}
pair = pairlist.getNext();
if (Thread.currentThread().isInterrupted()) {
throw new RuntimeException("interrupt after getNext");
}
if (pair == null) {
pairlist.update();
continue;
}
if (logger.isDebugEnabled()) {
logger.debug("pi = " + pair.pi);
logger.debug("pj = " + pair.pj);
}
S = red.SPolynomial(pair.pi, pair.pj);
if (S.isZERO()) {
pairlist.record(pair, S);
continue;
}
if (logger.isDebugEnabled()) {
logger.debug("ht(S) = " + S.leadingExpVector());
}
H = red.normalform(G, S); //mod
reduction++;
if (H.isZERO()) {
pairlist.record(pair, H);
continue;
}
if (logger.isDebugEnabled()) {
logger.debug("ht(H) = " + H.leadingExpVector());
}
H = H.monic();
// System.out.println("H = " + H);
if (H.isONE()) {
// pairlist.update( pair, H );
pairlist.putOne(); // not really required
synchronized (G) {
G.clear();
G.add(H);
}
fin.allIdle();
return;
}
if (logger.isDebugEnabled()) {
logger.debug("H = " + H);
}
synchronized (G) {
G.add(H);
}
pairlist.update(pair, H);
//pairlist.record( pair, H );
//pairlist.update();
}
logger.info("terminated, done " + reduction + " reductions");
}
}
/**
* Reducing worker threads for minimal GB.
*/
class MiReducerSeqPair> implements Runnable {
private final List> G;
private GenPolynomial H;
private final ReductionPar red;
private final Semaphore done = new Semaphore(0);
private static final Logger logger = Logger.getLogger(MiReducerSeqPair.class);
MiReducerSeqPair(List> G, GenPolynomial p) {
this.G = G;
H = p;
red = new ReductionPar();
}
/**
* to string
*/
@Override
public String toString() {
return "MiReducerSeqpair";
}
/**
* getNF. Blocks until the normal form is computed.
* @return the computed normal form.
*/
public GenPolynomial getNF() {
try {
done.acquire(); //done.P();
} catch (InterruptedException e) {
throw new RuntimeException("interrupt in getNF");
}
return H;
}
public void run() {
if (logger.isDebugEnabled()) {
logger.debug("ht(H) = " + H.leadingExpVector());
}
try {
H = red.normalform(G, H); //mod
done.release(); //done.V();
} catch (RuntimeException e) {
Thread.currentThread().interrupt();
//throw new RuntimeException("interrupt in getNF");
}
if (logger.isDebugEnabled()) {
logger.debug("ht(H) = " + H.leadingExpVector());
}
// H = H.monic();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy