[GiNaC-devel] GiNaC is not a C compiler [Was: Optimized C-style output]
Sheplyakov Alexei
varg at theor.jinr.ru
Mon Mar 26 08:02:22 CEST 2007
Dear Stefan,
On Thu, Mar 22, 2007 at 03:26:40PM +0100, Stefan Weinzierl wrote:
>
> thinking about it, we probably want the following: a data structure
> consisting of two lists,
> lhs = lst(t0,t1,...)
> rhs = lst(expr0,expr1,...)
> with the understanding that this should represent something like
> ...
> t1 = expr1;
> t0 = expr0;
There is a type which represents such objects, it is called exmap
(std::map<ex, ex_is_less>). Is new type really necessary?
> There is a consistency condition which needs to be fulfilled: The right
> hand side of the (j+1)-th entry cannot refer to symbols t0,...,tj, which
> would be defined later.
>
> One can consider two such structures equivalent, if the values of
> t0,...,tn would be the same after execution of the corresponding code
> fragments. In most cases we are just interested in n=0.
> This is actually an equivalence relation in the mathematical sense.
>
> For this structure we can have methods, which transform the structure into
> an equivalent one.
> These methods can add or remove entries from the two lists, or change the
> order.
> Methods, which just transform the rhs are in some sense trivial and not
> considered here.
>
> Examples of such methods could be:
> - replace common subexpressions of type T1, T2, ... in the rhs.
> - split containers with more than n_max arguments
> - (eliminate redundant variables)
>
> This structure has then a print method, which simply prints
> ...
> double t1 = expr1;
> double t0 = expr0;
>
> Everybody happy with this concept ?
No. There are 3 different _independent_ transformations: 1) finding
commong subexpressions, 2) splitting an expression into a smaller parts,
3) generating code. Each of them is useful on its own, so it would be nice
to keep them separate.
E.g.
/**
* Attempt to find common subexpressions and replace them with symbols
*/
const ex find_common_subex(const ex& e, exmap& repls);
I've already described suggested semantics in the previous mail.
/**
* Split large sums and products in @e with number of operands bigger
* then @threshold, replace obtained subexpressions with symbols
*/
const ex split_large_subex(const ex& e, exmap& repls, const size_t threshold);
Example:
ex e = x1+...+x1000;
exmap repls;
e = split_large_subex(e, repls, 500);
cout << e << endl;
// Should print, for instance:
// symbol100 + symbol101
for (exmap::const_iterator i=repls.begin(); i!=repls.end(); ++i)
cout << i->first << " => " << i->second << endl;
// should be something like
// symbol100 => x1+...+x500
// symbol101 => x501+...+x1000
void generate_code(const& ex e, const exmap& repls, std::ostream& os);
Example:
generate_code(e, repls, std::cout);
should print
const double f(const double[] x) {
double symbol100 = x[0]+...+x[499];
double symbol101 = x[500]+...+x[999];
double ret = symbol100 + symbol101;
return ret;
}
N.B. I don't insist on this particular kind of interface, I just give
examples to clarify what I mean. You are wellcome to suggest something
better.
Best regards,
Alexei
--
All science is either physics or stamp collecting.
-------------- next part --------------
A non-text attachment was scrubbed...
Name: signature.asc
Type: application/pgp-signature
Size: 827 bytes
Desc: Digital signature
URL: <http://www.ginac.de/pipermail/ginac-devel/attachments/20070326/3e07909f/attachment.sig>
More information about the GiNaC-devel
mailing list