123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329 |
- on echo;
- % This file can be run to turn bits of the REDUCE source code
- % into C so that this C can be compiled and linked in to make a
- % customised CSL executable that will red REDUCE faster.
- %
- % Run this using slowr38.img to select code to compile into C.
- % The functions to be compiled are extracted from a file "profile.dat"
- % that was created by "profile.red".
- %
- symbolic;
- % Three major parameters are available:
- %
- % fnames a list of files to create. Making the list longer (or
- % shorter) changes the amount of C that can be created.
- % The CSL source code has to know how many there are, and
- % its current default is for 12 files.
- %
- % size_per_file
- % this guides the compiler about how much to put in each
- % generated file, where the value 7000 results in each
- % file of generated C being in the range 120 to 150 Kbytes.
- %
- % force_count indicates how many functions from alg.tst statistics should
- % be included before anything else.
- %
- %
- % Also if "how_many" is set then this will limit the total number of
- % functions that are compiled into C. Since I expect to pass that via a
- % command line "-dhow_many=362" etc I allow for it being a string
- % not a number to start with. In ordinary circumstances this will not be
- % used, however it has proved INVALUABLE when tracking down cases where
- % compilation into C causes changes in behaviour... how_many can be used
- % with a binary-chop selection process to discover exactly which function
- % causes upset when compiled into C. Of course in release quality code I
- % hope there are no such cases!
- global '(fnames size_per_file force_count how_many);
- fnames := '("u01" "u02" "u03" "u04" "u05"
- "u06" "u07" "u08" "u09" "u10"
- "u11" "u12");
- size_per_file := 7000;
- force_count := 350;
- if not boundp 'how_many then how_many := 1000000
- else << how_many := compress explodec how_many;
- if not numberp how_many then how_many := 1000000 >>;
- on comp;
- % First update the patch information (if needbe).
- load!-module 'remake;
- % ensure_patches_are_up_to_date(); % NO LONGER DONE HERE
- % Here I need to consider the issue of patches. First consider patches that
- % had been in force when "profile.red" was run. In such cases a patched
- % function f1 has an associated replacement f1_123456789 (the numeric suffix
- % is a checksum on the new definition) and when the profile job was run
- % this replacement will have had its definition copied to f1. The way in
- % which CSL's mapstore function extracts counts will mean that the
- % thing in profile.dat relate to f1_123456789.
- % Usually things in profile.dat are in the form
- % (function_name . checksum_of_definition)
- % but for these patched things I will instead record
- % (original_function_name package_involved)
- % This can be distinguished because it has a symbol not a number as
- % the second component. To make this possible each patch function
- % f1_123456789 would have to have this information attached to it
- % when the profiling job was run.
- %
- % But I suppose have now obtained a newer version of the patches file. So
- % now the correct patch for f1 will be f1_abcdef. If f1 was in one of the
- % REDUCE core packages (eg "alg") then both the functions f1_123456789 and
- % f1_abcdef will be in memory now, but it will be the latter that will
- % have been copied to plain old f1. In other cases f1_123456789 will now
- % have been totally lost and the definition of f1_abcdef will be in the
- % patches module. Furthermore the new patches file may patch another
- % function f2 that had not previously been subject to patching, but
- % that had been selected for compilation into C. And in a truly bad
- % case the complete REDUCE sources will contain several functions named
- % f2 and of course the patches file identifies which one it is interested
- % in by the name of the package it is in.
- %
- % The response to all this I will use here is intended to make life
- % reasonably SIMPLE for me in a complicated situation. So I first
- % collect the set of names that I think need compiling into C. Then I
- % grab a list of the names of things defined in the current patches file.
- % If a function in the paches file has a name similar enough (!) to one that
- % I have already decided to compile into C then I will schedule it for
- % compilation into C too. Because of the hash suffix added to names in the
- % patches file defining a C version having those things present in the Lisp
- % kernel should never be a problem - after all the patches file itself is
- % intended to be loaded all the time. So the main down-side of this is
- % that I will sometimes find that I have compiled into C either patch
- % versions of a function when it was another version of that code that was
- % time-critical or that I have compiled into C two generations of
- % patch function. These waste opportunity and space by having some
- % things compiled into C that might not really justify that, but this
- % seems a modest cost.
- fluid '(w_reduce requests);
- w_reduce := requests := nil;
- % I make a list of all the functions that profile data suggests that
- % I should compile into C. The master copy of the profile data is
- % usually expected to be in "$srcdir/../csl-c", but I allow a copy in the
- % current directory (which is where the profiling process leaves it).
- symbolic procedure read_profile_data file;
- begin
- scalar w0, w1;
- if not errorp(w0 := errorset(list('open, file, ''input), nil, nil)) then <<
- w0 := rds car w0;
- while not errorp (w1 := errorset('(read), nil, nil)) and
- not eqcar(w1, !$eof!$) do
- requests := car w1 . requests;
- % The data structure read in here will be of the form
- % ((module-name f-name1 f_name2 ...) (module-name ...) ...)
- % where within each module the requested functions have been listed in
- % order of priority.
- close rds w0 >>
- end;
- % I read from the current directory only if I do not find anything
- % in the csl-c one.
- off echo;
- read_profile_data "$srcdir/../csl-c/profile.dat";
- if null requests then read_profile_data "profile.dat";
- on echo;
- % As a fairly shameless hack I am going to insist on compiling ALL the
- % things that the "alg" test uses. That is because this one test
- % fiel has been used for many years to give a single performance
- % figure for REDUCE. In fact it is not too bad to pay lots of
- % attention to it since it exercises the basic core algebra and so what is
- % good for it is good for quite a lot of everybody else. However by
- % tuning this selection process you can adjust the way REDUCE balances
- % its speed in different application areas.
- w_reduce := assoc('alg, requests)$
- requests := for each x in delete(w_reduce, requests) collect cdr x$
- w_reduce := reverse cdr w_reduce$
- d := length w_reduce - force_count;
- if d > 0 then for i := 1:d do w_reduce := cdr w_reduce;
- length w_reduce;
- % Now I will merge in suggestions from all other modules in
- % breadth-first order of priority
- % Ie if I have modules A, B, C and D (with A=alg) and each has in it
- % functions a1, a2, a3 ... (in priority odder) then I will make up a list
- % here that goes
- %
- % a1 a2 a3 ... an b1 c1 d2 b2 c2 d2 b3 c3 d3 b4 c4 d4 ...
- %
- % so that the first n items from A get priority and after that B, C and D
- % will get about balanced treatment if I have to truncate the list at
- % some stage.
- symbolic procedure membercar(a, l);
- if null l then nil
- else if a = caar l then t
- else membercar(a, cdr l);
- fg := t;
- while fg do <<
- fg := nil;
- for each x on requests do
- if car x then <<
- if k := assoc(caaar x, w_reduce) then <<
- if not (cadr k = cadaar x) then <<
- prin caaar x; printc " has multiple definition";
- princ " keep version with checksum: "; print cadr k;
- princ " ignore: "; print cadaar x;
- terpri() >> >>
- % ORDP is a special case because I have put a version of it into the
- % CSL kernel by hand, and any redefinition here would be unfriendly and
- % might clash with that.
- else if caaar x = 'ordp then printc "Ignoring ORDP (!)"
- else w_reduce := caar x . w_reduce;
- fg := t;
- rplaca(x, cdar x) >> >>;
- % Now I scan all pre-compiled modules to recover source versions of the
- % selected REDUCE functions. The values put as load!-source properties
- % are checksums of the recovered definitions that I would be prepared
- % to accept.
- for each n in w_reduce do put(car n, 'load!-source, cdr n);
- w_reduce := for each n in w_reduce collect car n$
- for each m in library!-members() do load!-source m;
- % Now deal with patches...
- load!-source := t;
- if modulep 'patches then patch!-functions := load!-source 'patches
- else patch!-functions := nil;
- % Some of the functions just collected are not patches for bits of REDUCE
- % but are the code that installs the patches. I do not worry too much
- % about that here.
- % Now I will scan down w_reduce (the list of all things to be compiled into C)
- % and if that contains an entry either f1 or f1_123456789 and there is
- % an entry f2_abcdef in the list of patch-functions then I will
- % insert f2_abcdef into the list of things to be compiled into C just
- % next to plain f2 or f2_123456789.
- %
- % The way I do this will often set up a few false-matches but the cost of
- % them is just that some unimportant things will get compiled into C.
- global '(tag!-chars);
- tag!-chars := explodec "0123456789abcdefghijklmnopqrstuvwxyz";
- symbolic procedure trim!-suffix name;
- begin
- scalar w;
- w := reverse explode name;
- if eqcar(w, '!_) then w := cdr w;
- if null w or not member(car w, tag!-chars) then return nil;
- w := cdr w;
- while w and member(car w, tag!-chars) do w := cdr w;
- if not eqcar(w, '!_) then return nil;
- w := cdr w;
- if null w then return nil
- else return compress reverse w
- end;
- w := w_reduce$
- w_reduce := nil;
- while w do <<
- w_reduce := car w . w_reduce;
- p := trim!-suffix car w;
- for each n in patch!-functions do
- if not (n = car w) and
- p and
- not (n member w_reduce) and
- p = trim!-suffix n then <<
- w_reduce := n . w_reduce;
- princ "+++ Also C-compile "; prin n; princ " as match for ";
- prin car w; princ ": root is "; print p >>;
- w := cdr w >>;
- verbos nil;
- global '(rprifn!*);
- on fastfor, fastvector, unsafecar;
- symbolic procedure listsize(x, n);
- if null x then n
- else if atom x then n+1
- else listsize(cdr x, listsize(car x, n+1));
- <<
- count := 0;
- while fnames do begin
- scalar name, bulk;
- name := car fnames;
- princ "About to create "; printc name;
- c!:ccompilestart(name, name, "$srcdir/../csl-c", nil);
- bulk := 0;
- while bulk < size_per_file and w_reduce and how_many > 0 do begin
- scalar name, defn;
- name := car w_reduce;
- if null (defn := get(name, '!*savedef)) then <<
- princ "+++ "; prin name; printc ": no saved definition found";
- w_reduce := cdr w_reduce >>
- else <<
- bulk := listsize(defn, bulk);
- if bulk < size_per_file then <<
- c!:ccmpout1 ('de . name . cdr defn);
- how_many := how_many - 1;
- count := count + 1;
- w_reduce := cdr w_reduce >> >> end;
- eval '(c!-end);
- fnames := cdr fnames
- end;
- terpri();
- printc "*** End of compilation from REDUCE into C ***";
- terpri();
- bulk := 0;
- % I list the next 50 functions that WOULD get selected - just for interest.
- if null w_reduce then printc "No more functions need compiling into C"
- else while bulk < 50 and w_reduce do
- begin
- name := car w_reduce;
- if null (defn := get(name, '!*savedef)) then <<
- princ "+++ "; prin name; printc ": no saved definition found";
- w_reduce := cdr w_reduce >>
- else <<
- bulk := bulk+1;
- print name;
- w_reduce := cdr w_reduce >> end;
- terpri();
- prin count; printc " functions compiled into C";
- nil >>;
- quit;
|