sclass AllOnAll<A, B> {
  new L<A> aList;
  new L<B> bList;
  LPair<IntRange> todo = new LinkedList;
  LPair<A, B> todo2 = new LinkedList;

  synchronized void newA(A a) {
    add(aList, a);
    addPair(todo, intRange_last(aList), intRange(0, l(bList)));
  }
  
  synchronized void newB(B b) {
    add(bList, b);
    addPair(todo, intRange(0, l(aList)), intRange_last(bList));
  }
  
  synchronized Pair<A, B> next() {
    while (empty(todo2)) {
      if (empty(todo)) null;
      Pair<IntRange> p = popFirst(todo);
      addAll(todo2, outerProduct_pairs(subList(aList, p.a), subList(bList, p.b)));
    }
    ret popFirst(todo2);
  }
}