From 48209a75a7fab2296abfb75ba6931fc5ee342472 Mon Sep 17 00:00:00 2001 From: rmkaplan Date: Sun, 21 Feb 2021 20:55:25 -0800 Subject: [PATCH] COMPAREDIRECTORIES: removed makesysout from MEDLEY-FIX-DIRS, new EOLTYPE tool Larry reorganized the directories so makesysout no longer exists. EOLTYPE now takes a SHOWCONTEXT argument. This prints the context of eol characters that are not consistent with the original type of the file. --- lispusers/COMPAREDIRECTORIES | 2 +- lispusers/COMPAREDIRECTORIES.LCOM | Bin 23248 -> 24355 bytes 2 files changed, 1 insertion(+), 1 deletion(-) diff --git a/lispusers/COMPAREDIRECTORIES b/lispusers/COMPAREDIRECTORIES index d204ff9e..169007cd 100644 --- a/lispusers/COMPAREDIRECTORIES +++ b/lispusers/COMPAREDIRECTORIES @@ -1 +1 @@ -(DEFINE-FILE-INFO PACKAGE "INTERLISP" READTABLE "INTERLISP" BASE 10) (FILECREATED " 7-Jan-2021 23:21:40"  {DSK}kaplan>Local>medley3.5>lispcore>lispusers>COMPAREDIRECTORIES.;275 63412 changes to%: (FNS COMPAREDIRECTORIES) previous date%: "31-Oct-2020 09:13:05" {DSK}kaplan>Local>medley3.5>lispcore>lispusers>COMPAREDIRECTORIES.;274) (* ; " Copyright (c) 1985, 1986, 1987, 1988, 1990, 1994, 1998, 2018, 2020, 2021 by Venue & Xerox Corporation. All rights reserved. ") (PRETTYCOMPRINT COMPAREDIRECTORIESCOMS) (RPAQQ COMPAREDIRECTORIESCOMS ( (* ;; "Compare the contents of two directories.") (FNS COMPAREDIRECTORIES CDFILES COMPAREDIRECTORIES.INFOS MATCHNAME) (FNS CDPRINT CDPRINT.LINE) (FNS CDMAP CDENTRY CDSUBSET) (FNS BINCOMP EOLTYPE) (RECORDS CDENTRY CDINFO) (* ;; "look for compiled files older than the sources") (FNS FIND-UNCOMPILED-FILES FIND-UNSOURCED-FILES FIND-SOURCE-FILES FIND-COMPILED-FILES FIND-UNLOADED-FILES FIND-LOADED-FILES FIND-MULTICOMPILED-FILES) (FNS CREATED-AS SOURCE-FOR-COMPILED-P COMPILE-SOURCE-DATE-DIFF) (FNS FIX-DIRECTORY-DATES FIX-EQUIV-DATES COPY-COMPARED-FILES COPY-MISSING-FILES COMPILED-ON-SAME-SOURCE) [VARS (ONESECOND (IDIFFERENCE (IDATE "1-Jan-2020 12:00:01") (IDATE "1-Jan-2020 12:00:00"] (INITVARS (LASTCDENTRIES NIL)) (COMS (FNS COMPARE-ENTRY-SOURCE-FILES) (FILES COMPARESOURCES)))) (* ;; "Compare the contents of two directories.") (DEFINEQ (COMPAREDIRECTORIES [LAMBDA (DIR1 DIR2 SELECT FILEPATTERNS EXTENSIONSTOAVOID USEDIRECTORYDATE OUTPUTFILE ALLVERSIONS) (* ; "Edited 7-Jan-2021 23:21 by rmk:") (* ;; "Compare the contents of two directories, e.g., for change-control purposes. Compares files matching FILEPATTERN (or *.*;) on DIR1 and DIR2, listing which is newer, or when one is not found on the other. If SELECT is or contains SAME/=, BEFORE/<, AFTER/>, then files where DIR1 is the same as, earlier than, or later than DIR2 are selected. SELECT= NIL is the same as (< >), T is the same as (< > =). Also allows selection based on file-length criteria.") (* ;; "") (* ;; "Unless USEDIRECTORYDATE, comparison is with respect to the the LISP filecreated dates if evailable.") (* ;; "") (* ;; "If OUTPUTFILE is NIL, the list of compared entries is returned. Otherwise the selected entries are printed on OUTPUTFILE (T for the display).") [SETQ SELECT (SELECTQ SELECT (NIL '(< > -* *-)) (T '(< > -* *- =)) (FOR S IN (MKLIST SELECT) COLLECT (SELECTQ S ((AFTER >) '>) ((BEFORE <) '<) ((SAME SAMEDATE =) '=) (AUTHOR 'AUTHOR) (-* '-*) (*- '*-) (~= '~=) (ERROR "UNRECOGNIZED SELECT PARAMETER" S] (PROG (INFOS1 INFOS2 CANDIDATES SELECTED COMPAREDATE DEPTH1 DEPTH2) [SETQ COMPAREDATE (INTERSECTION SELECT '(< > =] (* ;; "DIRECTORYNAME here to get unrelativized specifications for header.") (* ;; "Allow all subdirectories if a directory ends in *, but peel it off for the resolution") (CL:WHEN (EQ '* (NTHCHAR DIR1 -1)) (SETQ DEPTH1 T) (SETQ DIR1 (SUBSTRING DIR1 1 -2))) (CL:WHEN (EQ '* (NTHCHAR DIR2 -1)) (SETQ DEPTH2 T) (SETQ DIR2 (SUBSTRING DIR2 1 -2))) (SETQ DIR1 (OR (DIRECTORYNAME (OR DIR1 T)) (ERROR "DIRECTORY DOES NOT EXIST" DIR1))) (SETQ DIR2 (OR (DIRECTORYNAME (OR DIR2 T)) (ERROR "DIRECTORY DOES NOT EXIST" DIR2))) (PRINTOUT T "Comparing " DIR1 6 "vs. " DIR2 T "as of " (DATE) " selecting " SELECT " ... ") (SETQ INFOS1 (COMPAREDIRECTORIES.INFOS (CDFILES DIR1 FILEPATTERNS EXTENSIONSTOAVOID ALLVERSIONS DEPTH1) USEDIRECTORYDATE)) (SETQ INFOS2 (COMPAREDIRECTORIES.INFOS (CDFILES DIR2 FILEPATTERNS EXTENSIONSTOAVOID ALLVERSIONS DEPTH2) USEDIRECTORYDATE)) (CL:UNLESS (AND INFOS2 INFOS1) (RETURN)) (* ;; "At this point the CAR of each info is the atomic match-name. Peel it off to produce candidate entries.") (* ;;  "Look through all of the I2's because multiple versions (if VERSIONS) have the same matchname") [SETQ CANDIDATES (FOR I1 IN INFOS1 JOIN (IF ALLVERSIONS THEN (OR (FOR I2 IN INFOS2 WHEN (EQ (CAR I2) (CAR I1)) COLLECT (LIST (CAR I1) (CDR I1) (CDR I2))) (CONS (LIST (CAR I1) (CDR I1) NIL))) ELSE (CONS (LIST (CAR I1) (CDR I1) (CDR (ASSOC (CAR I1) INFOS2] (* ;; "Could be some 2's without 1's") (SORT [NCONC CANDIDATES (FOR I2 IN INFOS2 UNLESS (ASSOC (CAR I2) CANDIDATES) COLLECT (LIST (CAR I2) NIL (CDR I2] T) (* ;; "CANDIDATES is now a sorted list of the form (matchname entry1 entry2) where an entry consists of (fullname date length author)") (* ;; "Do the SELECT filtering and insert the date relation.") [SETQ SELECTED (FOR C MATCHNAME INFO1 INFO2 IDATE1 IDATE2 DATEREL BINCOMP IN CANDIDATES EACHTIME (SETQ MATCHNAME (POP C)) (SETQ INFO1 (POP C)) (SETQ INFO2 (POP C)) (IF (AND INFO1 INFO2) THEN (SETQ IDATE1 (IDATE (FETCH DATE OF INFO1))) (SETQ IDATE2 (IDATE (FETCH DATE OF INFO2))) (SETQ DATEREL (IF (IGREATERP IDATE1 IDATE2) THEN '> ELSEIF (ILESSP IDATE1 IDATE2) THEN '< ELSE '=)) ELSE (* ;; "Just for printing--no comparison") (SETQ DATEREL '*)) WHEN (IF (AND INFO1 INFO2) THEN (CL:WHEN (OR (NULL COMPAREDATE) (SELECTQ DATEREL (> (MEMB '> SELECT)) (< (MEMB '< SELECT)) (= (MEMB '= SELECT)) (SHOULDNT))) (SETQ BINCOMP (BINCOMP (FETCH FULLNAME OF INFO1) (FETCH FULLNAME OF INFO2) T (FETCH EOL OF INFO1) (FETCH EOL OF INFO2))) (* ;; "We want the ~= test to reflect exact byte equivalence, including the same EOL. We use the BINCOMP value below to indicate EOL differences, so we check it here.") [NOT (AND (MEMB '~= SELECT) BINCOMP (EQ (FETCH EOL OF INFO1) (FETCH EOL OF INFO2]) ELSEIF INFO1 THEN (* ;; "OK if INFO2 is missing?") (MEMB '*- SELECT) ELSE (* ;; "OK if INFO1 is missing?") (MEMB '-* SELECT)) COLLECT (CREATE CDENTRY MATCHNAME _ MATCHNAME INFO1 _ INFO1 DATEREL _ DATEREL INFO2 _ INFO2 EQUIV _ (CL:UNLESS (EQ DATEREL '*) BINCOMP] (PRINTOUT T (LENGTH SELECTED) " entries" T) (PUSH SELECTED (LIST DIR1 DIR2 SELECT (DATE))) (SETQ LASTCDENTRIES SELECTED) (CL:UNLESS OUTPUTFILE (RETURN SELECTED)) (RETURN (CDPRINT SELECTED OUTPUTFILE (MEMB 'AUTHOR SELECT SELECT]) (CDFILES [LAMBDA (DIR FILEPATTERNS EXTENSIONSTOAVOID ALLVERSIONS DEPTH) (* ; "Edited 16-Oct-2020 13:42 by rmk:") (* ;; "Returns a list of fullnames for files that satisfy the criteria") (* ;; "For each name returned by (DIRECTORY DIR), assumes that FILEPATTERNS applies to the suffix after the directory (i.e. after NAMEPOS). That includes possibly subdirectories, dotted files in ultimate file names, and versions.") (* ;; " Exclude subdirectories unless FILEPATTERNS includes *>*") (* ;; " Exclude dotted files (.xxx) unless FILEPATTERNS includes .*") (* ;; " Exclude files with extensions in EXTENSIONSTOAVOID (*=NIL does no filtering)") (* ;; " Exclude older versions unless ALLVERSIONS=T") (* ;; " DEPTH is the number of subdirectories below the ones specified in DIR (NIL top-level of DIR only, T = any depth)") (* ;; "Resolve relative directories, so we can suppress subdirectory matches. ") (SETQ EXTENSIONSTOAVOID (MKLIST (U-CASE EXTENSIONSTOAVOID))) [SETQ FILEPATTERNS (MKLIST (OR FILEPATTERNS '*] (FOR FP FN FPNAME FPEXT EXCLUDEDOTTED (TOPDIR _ (DIRECTORYNAME (OR DIR T))) IN FILEPATTERNS JOIN [SETQ FPNAME (U-CASE (FILENAMEFIELD FP 'NAME] [SETQ FPEXT (U-CASE (FILENAMEFIELD FP 'EXTENSION] (CL:UNLESS FPNAME (IF FPEXT THEN (* ;; ".XY") (SETQ FPNAME (PACK* "." FPEXT)) ELSE (SETQ FPNAME '*))) (CL:UNLESS FPEXT (SETQ FPEXT '*)) (SETQ EXCLUDEDOTTED (NEQ (CHARCODE %.) (CHCON1 FPNAME))) (SETQ FN (PACKFILENAME.STRING 'VERSION (CL:IF ALLVERSIONS '* "") 'DIRECTORY TOPDIR 'NAME '* 'EXTENSION '*)) (* ;; "DEPTH is the number of internal %">%"") [IF (EQ DEPTH T) THEN (SETQ DEPTH MAX.SMALLP) ELSEIF DEPTH ELSE (SETQ DEPTH (BIND (CNT _ 0) (POS _ 0) (FNDIR _ (FILENAMEFIELD FN 'DIRECTORY)) WHILE (SETQ POS (STRPOS ">" FNDIR (ADD1 POS))) DO (ADD CNT 1) FINALLY (RETURN CNT] (FOR FULLNAME NAME EXT THISDEPTH IN (DIRECTORY FN) EACHTIME [SETQ NAME (U-CASE (FILENAMEFIELD FULLNAME 'NAME] [SETQ EXT (U-CASE (FILENAMEFIELD FULLNAME 'EXTENSION] (CL:UNLESS NAME (IF EXT THEN (* ;; ".XY") (SETQ NAME (PACK* "." EXT)) (SETQ EXT NIL))) (CL:WHEN (AND EXCLUDEDOTTED (EQ (CHARCODE %.) (CHCON1 NAME))) (GO $$ITERATE)) (SETQ THISDEPTH (BIND (CNT _ 0) (POS _ 0) (FNDIR _ (FILENAMEFIELD FULLNAME 'DIRECTORY)) WHILE (SETQ POS (STRPOS ">" FNDIR (ADD1 POS))) DO (ADD CNT 1) FINALLY (RETURN CNT))) (* ;; "An empty subdirectory may appear without name or extensions") WHEN (AND (OR NAME EXT) (OR (EQ FPNAME '*) (EQ FPNAME NAME)) (OR (EQ FPEXT '*) (EQ FPEXT EXT))) UNLESS [OR (IGREATERP THISDEPTH DEPTH) (AND EXT (OR (MEMB '* EXTENSIONSTOAVOID) (MEMB EXT EXTENSIONSTOAVOID] COLLECT FULLNAME) FINALLY (CL:UNLESS $$VAL (PRINTOUT T "No relevant files in " TOPDIR T]) (COMPAREDIRECTORIES.INFOS [LAMBDA (FILES USEDIRECTORYDATE) (* ; "Edited 13-Oct-2020 08:42 by rmk:") (* ;; "Value is a list of CDINFOS with the match-name consed on to the front") (FOR FULLNAME TYPE LDATE IN FILES COLLECT (* ;; "GDATE/IDATE in case Y2K") (SETQ LDATE (FILEDATE FULLNAME)) (* ; "Is it a Lisp file?") (CONS (MATCHNAME FULLNAME) (CREATE CDINFO FULLNAME _ FULLNAME DATE _ [GDATE (IDATE (IF USEDIRECTORYDATE THEN (GETFILEINFO FULLNAME 'CREATIONDATE) ELSEIF (OR LDATE (GETFILEINFO FULLNAME 'CREATIONDATE] LENGTH _ (GETFILEINFO FULLNAME 'LENGTH) AUTHOR _ (GETFILEINFO FULLNAME 'AUTHOR) TYPE _ (IF LDATE THEN (CL:IF (MEMB (FILENAMEFIELD FULLNAME 'EXTENSION) *COMPILED-EXTENSIONS*) 'COMPILED 'SOURCE) ELSE (PRINTFILETYPE FULLNAME)) EOL _ (EOLTYPE FULLNAME]) (MATCHNAME [LAMBDA (NAME) (* ; "Edited 5-Sep-2020 13:41 by rmk:") (* ;; "The NAME.DIR for matching related files") (LET ((M (PACKFILENAME 'HOST NIL 'VERSION NIL 'DIRECTORY NIL 'BODY NAME))) (* ;; "Strip off the nuisance period") (CL:IF (EQ (CHARCODE %.) (NTHCHARCODE M -1)) (SUBATOM M 1 -2) M)]) ) (DEFINEQ (CDPRINT [LAMBDA (CDENTRIES FILE PRINTAUTHOR) (* ; "Edited 13-Oct-2020 08:38 by rmk:") (* ;; "Typically CDENTRIES will have a header. If not, we fake one up, at least for the directories and today's date.") (CL:UNLESS CDENTRIES (PRINTOUT T T "Note: Using LASTCDENTRIES" T T) (SETQ CDENTRIES LASTCDENTRIES)) (RESETLST (LET (INFO1 TEXT STREAM DATE1POS ENDDATE1 DIR1 DIR2 (HEADER (CAR CDENTRIES)) NCHARSDIR1) (CL:UNLESS (STRINGP (CADR HEADER)) (SETQ HEADER (LIST [FOR E IN CDENTRIES WHEN (FETCH INFO1 OF E) DO (RETURN (PACKFILENAME 'NAME NIL 'EXTENSION NIL 'VERSION NIL 'BODY (FETCH FULLNAME OF (FETCH INFO1 OF E] [FOR E IN CDENTRIES WHEN (FETCH INFO2 OF E) DO (RETURN (PACKFILENAME 'NAME NIL 'EXTENSION NIL 'VERSION NIL 'BODY (FETCH FULLNAME OF (FETCH INFO2 OF E] NIL (DATE))) (PUSH CDENTRIES HEADER)) (SETQ DIR1 (CAR HEADER)) (SETQ NCHARSDIR1 (NCHARS DIR1)) (SETQ DIR2 (CADR HEADER)) (CL:UNLESS (SETQ STREAM (GETSTREAM FILE 'OUTPUT T)) [RESETSAVE (SETQ STREAM (OPENSTREAM (PACKFILENAME 'EXTENSION 'TXT 'BODY FILE) 'OUTPUT 'NEW)) '(PROGN (CLOSEF? OLDVALUE]) (CL:WHEN DIR1 (PRINTOUT STREAM "Comparing " DIR1 6 "vs. " DIR2 T "as of " (CADDDR HEADER)) (CL:WHEN (CADDR HEADER) (PRINTOUT STREAM " selecting " (CADDR HEADER))) (PRINTOUT STREAM -2 (LENGTH (CDR CDENTRIES)) " entries" T T)) (LINELENGTH 1000 STREAM) (* ; "Don't wrap") (* ;; "DATE1POS is the position of the first character of INFO1's date, used for tabbing. We have to measure the filename, date, size, and author if desired") (IF (CDR CDENTRIES) THEN (FOR E INFO1 (MAXDATE1WIDTH _ 0) (SPACEWIDTH _ 1) (PARENWIDTH _ 2) IN (CDR CDENTRIES) WHEN (SETQ INFO1 (FETCH INFO1 OF E)) LARGEST [SETQ MAXDATE1WIDTH (IMAX MAXDATE1WIDTH (NCHARS (FETCH DATE OF INFO1] (IPLUS (- (NCHARS (FETCH FULLNAME OF INFO1)) NCHARSDIR1) (NCHARS (FETCH LENGTH OF INFO1)) (CL:IF PRINTAUTHOR (IPLUS SPACEWIDTH PARENWIDTH (NCHARS (FETCH AUTHOR OF INFO1))) 0)) FINALLY (* ;;  "First 4 for width of equiv. $$EXTREME is NIL if there are no INFO1's") (SETQ DATE1POS (IPLUS (OR $$EXTREME 10) 4 (ITIMES 3 SPACEWIDTH))) (SETQ ENDDATE1 (IPLUS DATE1POS MAXDATE1WIDTH) )) (FOR E IN (CDR CDENTRIES) DO (CDPRINT.LINE STREAM E PRINTAUTHOR DATE1POS ENDDATE1 NCHARSDIR1 (NCHARS DIR2))) ELSE (PRINTOUT T "CDENTRIES is empty" T)) (AND STREAM (CLOSEF? STREAM))))]) (CDPRINT.LINE [LAMBDA (STREAM ENTRY PRINTAUTHOR DATE1POS ENDDATE1 NCHARSDIR1 NCHARSDIR2) (* ; "Edited 13-Oct-2020 08:51 by rmk:") (* ;; "Format one line of the directory comparison listing. If PRINTAUTHOR and AUTHOR1 or AUTHOR2 are non-NIL, list the author in parens; otherwise omit it.") (LET ((INFO1 (FETCH INFO1 OF ENTRY)) (INFO2 (FETCH INFO2 OF ENTRY))) (PRINTOUT STREAM (SELECTQ (FETCH EQUIV OF ENTRY) (T "==") (NIL " ") (PROGN (SELECTQ (FETCH EOL OF INFO1) (CR 'C) (LF 'L) (CRLF 2) " ") (SELECTQ (FETCH EOL OF INFO2) (CR 'C) (LF 'L) (CRLF 2) " "))) " ") (CL:WHEN INFO1 (PRINTOUT STREAM (SUBSTRING (FETCH FULLNAME OF INFO1) (ADD1 NCHARSDIR1) NIL (CONSTANT (CONCAT))) " ") (CL:WHEN PRINTAUTHOR (PRINTOUT STREAM "(" (FETCH AUTHOR OF INFO1) ") ")) (PRINTOUT STREAM (FETCH LENGTH OF INFO1) .TAB0 DATE1POS (FETCH DATE OF INFO1))) (PRINTOUT STREAM .TAB0 ENDDATE1 " " (FETCH DATEREL OF ENTRY) " ") (CL:WHEN INFO2 (PRINTOUT STREAM (FETCH DATE OF INFO2) " " (SUBSTRING (FETCH FULLNAME OF INFO2) (ADD1 NCHARSDIR2) NIL (CONSTANT (CONCAT))) " ") (CL:WHEN PRINTAUTHOR (PRINTOUT STREAM "(" (FETCH AUTHOR OF INFO2) ") ")) (PRINTOUT STREAM (FETCH LENGTH OF INFO2))) (TERPRI STREAM]) ) (DEFINEQ (CDMAP [LAMBDA (CDENTRIES FN) (* ; "Edited 6-Sep-2020 15:58 by rmk:") (CL:UNLESS CDENTRIES (PRINTOUT T T "Note: Using LASTCDENTRIES" T T) (SETQ CDENTRIES LASTCDENTRIES)) (FOR CDE MATCHNAME INFO1 DATEREL INFO2 EQUIV IN (CDR CDENTRIES) DECLARE (SPECVARS MATCHNAME INFO1 DATEREL INFO2 EQUIV) EACHTIME (SETQ MATCHNAME (FETCH MATCHNAME OF CDE)) (SETQ INFO1 (FETCH INFO1 OF CDE)) (SETQ DATEREL (FETCH DATEREL OF CDE)) (SETQ INFO2 (FETCH INFO2 OF CDE)) (SETQ EQUIV (FETCH EQUIV OF CDE)) DO (APPLY* FN CDE]) (CDENTRY [LAMBDA (MATCHNAME CDENTRIES) (* ; "Edited 5-Sep-2020 21:09 by rmk:") (ASSOC MATCHNAME (OR CDENTRIES LASTCDENTRIES]) (CDSUBSET [LAMBDA (CDENTRIES FN) (* ; "Edited 15-Sep-2020 13:49 by rmk:") (CL:UNLESS CDENTRIES (PRINTOUT T T "Note: Using LASTCDENTRIES" T T) (SETQ CDENTRIES LASTCDENTRIES)) (CONS (CAR CDENTRIES) (FOR CDE MATCHNAME INFO1 DATEREL INFO2 EQUIV IN (CDR CDENTRIES) DECLARE (SPECVARS MATCHNAME INFO1 DATEREL INFO2 EQUIV) EACHTIME (SETQ MATCHNAME (FETCH MATCHNAME OF CDE)) (SETQ INFO1 (FETCH INFO1 OF CDE)) (SETQ DATEREL (FETCH DATEREL OF CDE)) (SETQ INFO2 (FETCH INFO2 OF CDE)) (SETQ EQUIV (FETCH EQUIV OF CDE)) WHEN (APPLY* FN CDE) COLLECT CDE]) ) (DEFINEQ (BINCOMP [LAMBDA (FILE1 FILE2 EOLDIFFOK EOL1 EOL2) (* ; "Edited 13-Oct-2020 08:53 by rmk:") (* ;; "Returns T if FILE1 and FILE2 are byte-equivalent. Returns EOLDIFF if they are byte equivalent except for CR/LF/CRLF exchanges. ") (* ;; "If EOLDIFFOK, return indicates that the files are the same except for EOL mappings. If EOL1 and EOL2 are not provided, they are computed here.") (IF (IEQP (GETFILEINFO FILE1 'LENGTH) (GETFILEINFO FILE2 'LENGTH)) THEN [CL:WITH-OPEN-FILE (STREAM1 FILE1 :DIRECTION :INPUT) (CL:WITH-OPEN-FILE (STREAM2 FILE2 :DIRECTION :INPUT) (SETFILEINFO STREAM1 'ENDOFSTREAMOP (FUNCTION NILL)) (* ;; "Simpler code to recompute eol's even if provided") (BIND B1 B2 EOL1 EOL2 EOLDIFF WHILE (SETQ B1 (\BIN STREAM1)) UNLESS (EQ B1 (SETQ B2 (\BIN STREAM2))) DO (CL:UNLESS (AND EOLDIFFOK (SELCHARQ B1 (CR (CL:WHEN (EQ EOL1 'LF) (RETURN NIL)) (SETQ EOL1 'CR) (SETQ EOL2 'LF) (EQ B2 (CHARCODE LF))) (LF (CL:WHEN (EQ EOL1 'CR) (RETURN NIL)) (SETQ EOL1 'LF) (SETQ EOL2 'CR) (EQ B2 (CHARCODE CR))) NIL)) (RETURN NIL)) (CL:UNLESS EOLDIFF (SETQ EOLDIFF (LIST EOL1 EOL2))) FINALLY (RETURN (OR EOLDIFF T] ELSEIF EOLDIFFOK THEN (* ;; "Lengths are different possibly because of CRLF to CR/LF substitutions.") (* ;;  "More complex code could detect the EOLTYPE incrementally without separate passes, but ...") (CL:UNLESS EOL1 (SETQ EOL1 (EOLTYPE FILE1))) (CL:UNLESS EOL2 (SETQ EOL2 (EOLTYPE FILE2))) (CL:WHEN (IF [AND (EQ EOL1 'CRLF) (MEMB EOL2 '(LF CR] ELSEIF [AND (EQ EOL2 'CRLF) (MEMB EOL1 '(LF CR] THEN (SWAP FILE1 FILE2)) (* ;; "FILE1 is now CRLF, FILE2 is not. If FILE1 isn't longer, it can't have a CRLF that corresponds to a CR or LF.") (CL:WHEN (IGREATERP (GETFILEINFO FILE1 'LENGTH) (GETFILEINFO FILE2 'LENGTH)) [CL:WITH-OPEN-FILE (STREAM1 FILE1 :DIRECTION :INPUT) (CL:WITH-OPEN-FILE (STREAM2 FILE2 :DIRECTION :INPUT) (SETFILEINFO STREAM1 'ENDOFSTREAMOP (FUNCTION NILL)) (BIND B1 B2 EOLDIFF WHILE (SETQ B1 (\BIN STREAM1)) UNLESS (EQ B1 (SETQ B2 (\BIN STREAM2))) DO (CL:UNLESS [AND (EQ (CHARCODE CR) B1) (EQ (CHARCODE LF) (\BIN STREAM1)) (MEMB B2 (CHARCODE (CR LF] (RETURN NIL)) (CL:UNLESS EOLDIFF (SETQ EOLDIFF (LIST EOL1 EOL2))) FINALLY (RETURN (OR EOLDIFF T]))]) (EOLTYPE [LAMBDA (FILE) (* ; "Edited 3-Sep-2020 17:05 by rmk:") (* ;; "Returns the EOLCONVENTION of FILE if it only sees one kind, NIL if it can't decide.") (CL:WITH-OPEN-FILE (STREAM FILE :DIRECTION :INPUT) (SETFILEINFO STREAM 'ENDOFSTREAMOP (FUNCTION NILL)) (BIND EOLTYPE DO (SELCHARQ (OR (\BIN STREAM) (RETURN EOLTYPE)) (CR (IF (EQ (CHARCODE LF) (\PEEKBIN STREAM T)) THEN (CL:WHEN (MEMB EOLTYPE '(LF CR)) (RETURN NIL)) (\BIN STREAM) (SETQ EOLTYPE 'CRLF) ELSEIF (MEMB EOLTYPE '(LF CRLF)) THEN (RETURN NIL) ELSE (SETQ EOLTYPE 'CR))) (LF (CL:WHEN (MEMB EOLTYPE '(CR CRLF)) (RETURN NIL)) (SETQ EOLTYPE 'LF)) NIL]) ) (DECLARE%: EVAL@COMPILE (RECORD CDENTRY (MATCHNAME INFO1 DATEREL INFO2 . EQUIV)) (RECORD CDINFO (FULLNAME DATE LENGTH AUTHOR TYPE EOL)) ) (* ;; "look for compiled files older than the sources") (DEFINEQ (FIND-UNCOMPILED-FILES [LAMBDA (FILES DFASLMARGIN COMPILEEXTS) (* ; "Edited 20-Sep-2020 23:04 by rmk:") (* ; "Edited 3-Nov-94 15:17 by jds") (* ;; "Produces a list of the source files in FILES that have no corresponding compiled file") (* ;; "This determines whether there is at least one compiled file. If there are two or more, that's a problem") (* ;; "We want the most recent version only") (* ;; "Source files have a 2-element created-as with a non-NIL date") (SETQ FILES (FOR F IN (OR (LISTP FILES) (FILDIR FILES)) UNLESS (MEMB (SETQ F (PACKFILENAME 'VERSION NIL 'BODY F)) $$VAL) COLLECT F)) (FOR F SCREATION FILES IN FILES WHEN (AND (CADR (SETQ SCREATION (CREATED-AS F))) (NOT (CDDR SCREATION))) WHEN [SETQ FILES (FOR CEXT CF IN (OR COMPILEEXTS *COMPILED-EXTENSIONS*) WHEN (SETQ CF (INFILEP (PACKFILENAME 'EXTENSION CEXT 'VERSION NIL 'BODY F))) COLLECT (CL:WHEN (SOURCE-FOR-COMPILED-P SCREATION CF DFASLMARGIN) (RETURN NIL)) CF FINALLY (* ;; "If we found some compiled files, they weren't on this source. If there weren't any compiled files to check, maybe there weren't any functions.") (* ;;  "NLSETQ because we don't want to stop if there is an error, typically from a package problem") (RETURN (OR $$VAL (LET [(FCOMS (CAR (NLSETQ (GETDEF (FILECOMS F) 'VARS F] (IF (NULL FCOMS) THEN (* ;;  "GETDEF caused an error. Maybe a package problem. ") (AND NIL 'NOCOMMANDS) ELSEIF (INFILECOMS? NIL '(FUNCTIONS FNS) FCOMS) THEN T] COLLECT (CONS F (SELECTQ FILES (T NIL) (NOCOMMANDS (CONS "No commands")) (FOR CF IN FILES COLLECT (* ;;  "Positive means that compiled is later than source, normal order but maybe by too much.") (* ;;  "Negative means that compiled came before source. Odd") (LIST CF (COMPILE-SOURCE-DATE-DIFF CF SCREATION]) (FIND-UNSOURCED-FILES [LAMBDA (FILES DFASLMARGIN COMPILEEXTS) (* ; "Edited 15-Sep-2020 15:32 by rmk:") (* ; "Edited 3-Nov-94 15:17 by jds") (* ;;  "Produces a list of compiled FILES for which no source file can be found in the same directory.") (* ;; "The source date in at least one DFASL was off by a second, maybe some sort of IDATE rounding? So, give a margin.") (* ;; "We want the most recent version only. Check CREATED-AS to make sure it really is a compiled file.") (* ;; "Sort to get lcoms and dfasls next to each other.") (LET (CCREATEDS) (SETQ CCREATEDS (FOR CEXT FOUND CCREATED INSIDE (OR COMPILEEXTS *COMPILED-EXTENSIONS*) JOIN (FOR CF IN [OR (LISTP FILES) (FILDIR (PACKFILENAME 'EXTENSION CEXT 'VERSION "" 'BODY '*] WHEN (CDDR (SETQ CCREATED (CREATED-AS CF))) UNLESS (MEMBER CCREATED $$VAL) COLLECT CCREATED))) (* ;; "CCREATEDS is now a list of CREATED-AS items") (FOR CC SF IN CCREATEDS UNLESS (AND [SETQ SF (INFILEP (PACKFILENAME 'EXTENSION NIL 'VERSION NIL 'BODY (CAR CC] (SOURCE-FOR-COMPILED-P (SETQ SF (CREATED-AS SF)) CC DFASLMARGIN)) COLLECT [LIST (CAR CC) (AND SF (LIST (CAR SF) (ROUND (COMPILE-SOURCE-DATE-DIFF CC SF] FINALLY (RETURN (SORT $$VAL (FUNCTION (LAMBDA (CF1 CF2) (ALPHORDER (FILENAMEFIELD (CAR CF1) 'NAME) (FILENAMEFIELD (CAR CF2) 'NAME]) (FIND-SOURCE-FILES [LAMBDA (CFILES SDIRS DFASLMARGIN) (* ; "Edited 9-Sep-2020 12:26 by rmk:") (* ;; "Returns (CFILE . SFILES) pairs where CFILE is a Lisp compiled file in CFILES SFILES is a list of source files in SDIRS that CFILE was compiled on.") (* ;; "This suggests that one of CFILES should be copied to the SFILE directory.") (SETQ SDIRS (FOR SD INSIDE (OR SDIRS T) COLLECT (DIRECTORYNAME SD))) (SORT (FOR CF SFILES CNAME CCREATED IN (OR (LISTP CFILES) (FILDIR CFILES)) WHEN (AND (SETQ CNAME (INFILEP CF)) (CDDR (SETQ CCREATED (CREATED-AS CF))) (SETQ SFILES (FOR SD SF IN SDIRS WHEN (AND (SETQ SF (INFILEP (PACKFILENAME 'NAME (FILENAMEFIELD CF 'NAME) 'BODY SD))) (SOURCE-FOR-COMPILED-P SF CCREATED DFASLMARGIN)) COLLECT SF))) COLLECT (CONS CNAME SFILES)) (FUNCTION (LAMBDA (P1 P2) (ALPHORDER (FILENAMEFIELD (CAR P1)) (FILENAMEFIELD (CAR P2]) (FIND-COMPILED-FILES [LAMBDA (SFILES CDIRS DFASLMARGIN) (* ; "Edited 9-Sep-2020 12:26 by rmk:") (* ;; "Returns (SFILE . CFILES) pairs where SFILE is a Lisp source file in SFILES CFILES is a list of compiled files in CDIRS that were compiled on SFILE.") (* ;; "FILEDATE is true for source files and compiled files") (* ;; "This suggests that one of CFILES should be copied to the SFILE directory.") (SETQ CDIRS (FOR CD INSIDE (OR CDIRS T) COLLECT (DIRECTORYNAME CD))) (SORT (FOR SF CFILES SNAME SCREATED IN (OR (LISTP SFILES) (FILDIR SFILES)) WHEN [AND (SETQ SNAME (INFILEP SF)) (SETQ SCREATED (CREATED-AS SF)) (NOT (CDDR SCREATED)) (SETQ CFILES (FOR CEXT (ROOT _ (FILENAMEFIELD SNAME 'NAME)) IN *COMPILED-EXTENSIONS* JOIN (FOR CD CF IN CDIRS WHEN (AND (SETQ CF (INFILEP (PACKFILENAME 'NAME ROOT 'EXTENSION CEXT 'BODY CD))) (SOURCE-FOR-COMPILED-P SCREATED CF DFASLMARGIN)) COLLECT CF] COLLECT (CONS SNAME CFILES )) (FUNCTION (LAMBDA (P1 P2) (ALPHORDER (FILENAMEFIELD (CAR P1)) (FILENAMEFIELD (CAR P2]) (FIND-UNLOADED-FILES [LAMBDA (FILES) (* ; "Edited 9-Sep-2020 19:35 by rmk:") (* ;; "Returns the files in FILES that don't have FILECREATED properties and presumably are therefore not loaded in the current sysout.") (FOR F IN (OR (LISTP FILES) (FILDIR FILES)) WHEN (AND (SETQ F (INFILEP (CL:IF (LISTP F) (CAR F) F))) (FILEDATE F)) UNLESS (GETP (FILENAMEFIELD F 'NAME) 'FILEDATES) COLLECT F]) (FIND-LOADED-FILES [LAMBDA (ROOTFILENAMES) (* ; "Edited 19-Sep-2020 07:20 by rmk:") (FOR RN INSIDE ROOTFILENAMES WHEN (GETP RN 'FILEDATES) COLLECT (CONS RN (FOR F IN LOADEDFILELST WHEN (EQ RN (FILENAMEFIELD F 'NAME)) COLLECT F]) (FIND-MULTICOMPILED-FILES [LAMBDA (FILES SHOWINFO) (* ; "Edited 20-Sep-2020 20:57 by rmk:") (* ;; "Returns a list of names for files in FILES that have multiple compilations") (LET (SFILES) (FOR F EXT NAME IN (OR (LISTP FILES) (FILDIR FILES)) WHEN (MEMB (SETQ EXT (FILENAMEFIELD F 'EXTENSION)) *COMPILED-EXTENSIONS*) DO (SETQ NAME (FILENAMEFIELD F 'NAME)) (* ;; "PUSHNEW because we haven't filtered out versions") (PUSHNEW [CDR (OR (ASSOC NAME SFILES) (CAR (PUSH SFILES (CONS NAME] EXT)) (FOR S IN SFILES WHEN (CDDR S) COLLECT (IF SHOWINFO THEN `[,(CAR S) ,(CADAR (FIND-LOADED-FILES (CAR S))) ,(CREATED-AS (CAR S)) ,@(FOR EXT IN (SORT (CDR S)) COLLECT (CREATED-AS (PACKFILENAME 'EXTENSION EXT 'BODY (CAR S] ELSE (CAR S]) ) (DEFINEQ (CREATED-AS [LAMBDA (FILE) (* ; "Edited 20-Sep-2020 23:06 by rmk:") (* ;; "For lisp source files, returns (filecreatename filecreateddate)") (* ;; "For lisp compiled files, returns (cfilename cfiledate sfilecreatename sfilecreateddate)") (* ;; "For other files, (fullfilename NIL)") (* ;; "The cfilename is just the current directory name for DFASLs.") (* ;; "So: (CADR value) is non-NIL for Lisp files. Of those, (CDDR value) is non-NIL for compiled files.") (* ;; "We disable the package delimiter because the atoms in changes may have a packages that we don't know.") (CL:WITH-OPEN-FILE (STREAM FILE :DIRECTION :INPUT) (LET (FILEDATE FILENAME SOURCEDATE SOURCENAME LINE POS) [IF (EQ (CHARCODE %() (SKIPSEPRCODES STREAM)) THEN (* ; "Managed source or LCOM") (RESETLST [LET (FORM SFORM (RDTBL (FIND-READTABLE "OLD-INTERLISP-FILE"))) (SETQ POS (GETFILEPTR STREAM)) (READCCODE STREAM) (IF (EQ 'DEFINE-FILE-INFO (RATOM STREAM RDTBL)) THEN (* ;; "Reading is package-safe") (SETFILEPTR STREAM POS) (SETQ FORM (READ STREAM RDTBL)) (SETQ RDTBL (FIND-READTABLE (LISTGET (CDR FORM) :READTABLE))) ELSE (SETFILEPTR STREAM POS)) (CL:WHEN (EQ 'PACKAGEDELIM (GETSYNTAX '%: RDTBL)) [RESETSAVE (SETSYNTAX '%: 'OTHER RDTBL) `(SETSYNTAX %: PACKAGEDELIM ,RDTBL]) (* ;; "One way or the other, we're ready for the filecreated") (CL:WHEN (EQ (CHARCODE %() (SKIPSEPRCODES STREAM)) (SETQ FORM (READ STREAM RDTBL)) (CL:WHEN (MEMB (U-CASE (CAR FORM)) '(FILECREATED IL%:FILECREATED)) (* ;; "IL%%:FILECREATED because we screwed the readtable.") (IF [STREQUAL "compiled on " (CAR (LISTP (CADDR FORM] THEN (* ; "LCOM, get source info") (IF [AND (EQ (CHARCODE %() (SKIPSEPRCODES STREAM)) (MEMB [U-CASE (CAR (SETQ SFORM (READ STREAM RDTBL] '(FILECREATED IL%:FILECREATED] THEN (SETQ FILENAME (FULLNAME STREAM)) (SETQ FILEDATE (CADR FORM)) (SETQ SOURCENAME (CADDR SFORM)) (SETQ SOURCEDATE (CADR SFORM)) ELSE (SETQ FILENAME (FULLNAME STREAM)) (SETQ FILEDATE (CADR FORM))) ELSE (SETQ FILENAME (CADDR FORM)) (SETQ FILEDATE (CADR FORM)))))]) ELSEIF (SETQ POS (STRPOS "XCL Compiler output for source file " (SETQ LINE (CL:READ-LINE STREAM)) 1 NIL NIL T)) THEN (* ; "DFASL compiled?") (SETQ SOURCENAME (SUBATOM LINE POS)) (CL:WHEN (SETQ POS (STRPOS "Source file created " (SETQ LINE (CL:READ-LINE STREAM)) 1 NIL NIL T)) [SETQ SOURCEDATE (GDATE (IDATE (SUBSTRING LINE POS] (CL:WHEN (SETQ POS (STRPOS "FASL file created " (SETQ LINE (CL:READ-LINE STREAM)) 1 NIL NIL T)) [SETQ FILEDATE (GDATE (IDATE (SUBSTRING LINE POS]))] (* ;; "Revert filenames to Interlisp package if needed:") (CL:WHEN (STRPOS "IL:" FILENAME) (SETQ FILENAME (SUBATOM FILENAME 4))) (CL:WHEN (STRPOS "IL:" SOURCENAME) (SETQ SOURCENAME (MKATOM SOURCENAME 4))) (* ;; "Return DATE NIL if file is not a Lisp file") `(,(OR FILENAME (FULLNAME STREAM)) ,(AND FILEDATE (GDATE (IDATE FILEDATE))) ,@(CL:WHEN SOURCENAME (LIST SOURCENAME (GDATE (IDATE SOURCEDATE))))]) (SOURCE-FOR-COMPILED-P [LAMBDA (SOURCE COMPILED DFASLMARGIN) (* ; "Edited 31-Oct-2020 09:12 by rmk:") (* ;; "There seems to be some variation between the source dates in dfasl files and the filecreated date in the sources, they often don't match exactly. But if they are within DFASLMARGIN, we assume a match. We require exact date match for LCOMS") (* ;; "This is needed for dfasl files created before they recorded the source filecreated name and date instead of the directory source name and date when compile took place.") (* ;; "") (* ;; "DFASLMARGIN is a pair (after before) where we assume a match if the compiled date is no more than after minutes after the source date and no more than before minuts before (the diff is negative then).") (* ;; "A single positive integer x is interpreted as (x 0). A single negative integer x is interpreted as (-x x) (before or after x).") (* ;; "Default is (20 0).") (* ;; "T is positive or negative infinity") (CL:UNLESS (LISTP SOURCE) (SETQ SOURCE (CREATED-AS SOURCE))) (CL:UNLESS (LISTP COMPILED) (SETQ COMPILED (CREATED-AS COMPILED))) (SETQ DFASLMARGIN (IF (NULL DFASLMARGIN) THEN (* ;;  "If compiled is later than source by less than 20 minutes, it's probably OK") '(20 0) ELSEIF (EQ T DFASLMARGIN) THEN '(T 0) ELSEIF (LISTP DFASLMARGIN) ELSEIF (NOT (FIXP DFASLMARGIN)) THEN (ERROR "ILLEGAL DFASLMARGIN" DFASLMARGIN) ELSEIF (MINUSP DFASLMARGIN) THEN (LIST (MINUS DFASLMARGIN) DFASLMARGIN) ELSE (LIST DFASLMARGIN 0))) (OR (EQUAL (CAR SOURCE) (CADDR COMPILED)) (EQUAL (CADR SOURCE) (CADDDR COMPILED)) (AND [EQ 'DFASL (U-CASE (FILENAMEFIELD (CAR COMPILED) 'EXTENSION] (LET ((TIMEDIFF (COMPILE-SOURCE-DATE-DIFF COMPILED SOURCE))) (* ;; "If compiled was no more than 20 minutes later, it's probably OK. Of no more than DFASLMARGIN earlier, if it is negative.") (AND (OR (EQ T (CAR DFASLMARGIN)) (LEQ TIMEDIFF (CAR DFASLMARGIN))) (OR (EQ T (CADR DFASLMARGIN)) (GEQ TIMEDIFF (CADR DFASLMARGIN]) (COMPILE-SOURCE-DATE-DIFF [LAMBDA (CFILE SFILE) (* ; "Edited 20-Sep-2020 22:59 by rmk:") (* ;; "Positive means that compiled is later than source, normal order but maybe by too much. Negative means that compiled came before source, i.e., compiled on a source that didn't yet exist.") (* ;; "Value is in minutes") (ROUND (FQUOTIENT [IDIFFERENCE [IDATE (CADDDR (OR (LISTP CFILE) (CREATED-AS CFILE] (IDATE (CADR (OR (LISTP SFILE) (CREATED-AS SFILE] (TIMES 60 ONESECOND]) ) (DEFINEQ (FIX-DIRECTORY-DATES [LAMBDA (FILES MARGIN) (* ; "Edited 30-Oct-2020 22:01 by rmk:") (* ;; "For Lisp source and compiled files, ensures that the directory file date corresponds to the filecreated date. Returns the list of files whose dates were changed.") (* ;; "This allows for the fact that directory dates that are no later than, say, 30 seconds of the filecreated date are probably OK--the directory date may be set when the file is closed.") (* ;; "Use IDATEs in case FDCDATE is not Y2K.") (* ;; "Stop if directory date is more than 2 minutes earlier than the filecreated date. Earlier could be because the dates are asserted at different points in the filing process. But 2 minutes is worth thinking about. Returning from HELP will get them aligned.") (SETQ MARGIN (ITIMES (OR MARGIN 2) 60 ONESECOND)) (FOR F DIDATE FCDATE IN (OR (LISTP FILES) (FILDIR FILES)) WHEN (SETQ FCDATE (FILEDATE F)) UNLESS (IEQP (SETQ DIDATE (GETFILEINFO F 'ICREATIONDATE)) (SETQ FCDATE (IDATE FCDATE))) COLLECT (CL:WHEN (IGREATERP (IDIFFERENCE FCDATE DIDATE) MARGIN) (HELP "DIRECTORY DATE EARLIER THAN FILECREATED DATE" (LIST F (GDATE DIDATE) (GDATE FCDATE)))) (SETFILEINFO F 'ICREATIONDATE FCDATE) F]) (FIX-EQUIV-DATES [LAMBDA (CDENTRIES) (* ; "Edited 1-Sep-2020 16:21 by rmk:") (* ;; "For every entry whose files are EQUIVALENT and whose filedates are different, sets the directory of the file with the later date to be the date of the one with the earlier date. This preumes that the later one must have been a copy. ") (CL:UNLESS CDENTRIES (PRINTOUT T "Note: Using LASTCDENTRIES" T) (SETQ CDENTRIES LASTCDENTRIES)) (FOR CDE EARLY LATE IN (CDR CDENTRIES) WHEN (FETCH EQUIV OF CDE) UNLESS (EQ '= (FETCH DATEREL OF CDE)) COLLECT (SELECTQ (FETCH DATEREL OF CDE) (> (SETQ EARLY (FETCH INFO2 OF CDE)) (SETQ LATE (FETCH INFO1 OF CDE))) (< (SETQ EARLY (FETCH INFO1 OF CDE)) (SETQ LATE (FETCH INFO2 OF CDE))) (SHOULDNT)) (SETFILEINFO (FETCH FULLNAME OF LATE) 'ICREATIONDATE (GETFILEINFO (FETCH FULLNAME OF EARLY) 'ICREATIONDATE)) (FETCH FULLNAME OF LATE]) (COPY-COMPARED-FILES [LAMBDA (CDENTRIES TARGET MATCHNAMES) (* ; "Edited 1-Sep-2020 16:20 by rmk:") (* ;; "Copies source files to target files whose matchname belongs to MATCHNAMES, if given.") (* ;; "TARGET is 1 or 2, indicating which side of the CD entry is the target. Value is the list of matchnames whose files have been copied.") (* ;; "Directory filedates and other properties are preserved.") (CL:UNLESS (MEMB TARGET '(1 2)) (ERROR "INVALID TARGET" TARGET)) (CL:UNLESS CDENTRIES (PRINTOUT T "Note: Using LASTCDENTRIES" T) (SETQ CDENTRIES LASTCDENTRIES)) (SETQ MATCHNAMES (MKLIST MATCHNAMES)) (FOR CDE SINFO TINFO MATCHNAME IN (CDR CDENTRIES) EACHTIME (SETQ SINFO (FETCH INFO1 OF CDE)) (SETQ TINFO (FETCH INFO2 OF CDE)) (CL:WHEN (EQ TARGET 1) (SWAP SINFO TINFO)) (SETQ MATCHNAME (FETCH MATCHNAME OF CDE)) WHEN (AND (FETCH FULLNAME OF SINFO) (FETCH FULLNAME OF TINFO)) UNLESS (AND MATCHNAMES (NOT (MEMB MATCHNAME MATCHNAMES))) COLLECT (COPYFILE (FETCH FULLNAME OF SINFO) (PACKFILENAME 'VERSION NIL 'BODY (FETCH FULLNAME OF TINFO))) MATCHNAME]) (COPY-MISSING-FILES [LAMBDA (CDENTRIES TARGET MATCHNAMES) (* ; "Edited 1-Sep-2020 16:21 by rmk:") (* ;; "Copies source files to target files whose matchname belongs to MATCHNAMES, if given.") (* ;; "TARGET is 1 or 2, indicating which side of the CD entry is the target. Value is the list of matchnames whose files have been copied.") (* ;; "Directory filedates and other properties are preserved.") (CL:UNLESS (MEMB TARGET '(1 2)) (ERROR "INVALID TARGET" TARGET)) (CL:UNLESS CDENTRIES (PRINTOUT T "Note: Using LASTCDENTRIES" T) (SETQ CDENTRIES LASTCDENTRIES)) (CL:UNLESS (STRINGP (CADR (CAR CDENTRIES))) (ERROR "(CAR CDENTRIES) IS NOT A VALID PARAMETER LIST" (CAR CDENTRIES))) (SETQ MATCHNAMES (MKLIST MATCHNAMES)) (FOR CDE SINFO TINFO TDIR MATCHNAME (TDIR _ (CL:IF (EQ TARGET 1) (CAAR CDENTRIES) (CADAR CDENTRIES))) IN (CDR CDENTRIES) EACHTIME (SETQ MATCHNAME (FETCH MATCHNAME OF CDE)) (SETQ SINFO (FETCH INFO1 OF CDE)) (SETQ TINFO (FETCH INFO2 OF CDE)) (CL:WHEN (EQ TARGET 1) (SWAP SINFO TINFO)) WHEN (AND (FETCH FULLNAME OF SINFO) (NOT (FETCH FULLNAME OF TINFO))) UNLESS (AND MATCHNAMES (NOT (MEMB MATCHNAME MATCHNAMES))) COLLECT (* ;; "Using the source fullname in the target should preserve the version number") (COPYFILE (FETCH FULLNAME OF SINFO) (PACKFILENAME 'BODY TDIR 'BODY (FETCH FULLNAME OF SINFO))) MATCHNAME]) (COMPILED-ON-SAME-SOURCE [LAMBDA (CDENTRIES) (* ; "Edited 9-Sep-2020 13:00 by rmk:") (* ;; "Returms a subset of CDENTRIES consisting of files that are compiled on the same source (i.e. their source names or dates are the same). Preserves the header.") (CDSUBSET CDENTRIES (FUNCTION (LAMBDA (CDE) (DECLARE (USEDFREE INFO1 INFO2)) (LET (CREATED1 CREATED2) (CL:WHEN [AND (EQ 'COMPILED (FETCH TYPE OF INFO1)) (EQ 'COMPILED (FETCH TYPE OF INFO2)) [CDDR (SETQ CREATED1 (CREATED-AS (FETCH FULLNAME OF INFO1] (CDDR (SETQ CREATED2 (CREATED-AS (FETCH FULLNAME OF INFO2] (OR (EQUAL (CADDR CREATED1) (CADDR CREATED2)) (EQUAL (CADDDR CREATED1) (CADDDR CREATED2))))]) ) (RPAQ ONESECOND (IDIFFERENCE (IDATE "1-Jan-2020 12:00:01") (IDATE "1-Jan-2020 12:00:00"))) (RPAQ? LASTCDENTRIES NIL) (DEFINEQ (COMPARE-ENTRY-SOURCE-FILES [LAMBDA (CDENTRY LISTSTREAM EXAMINE DW?) (* ; "Edited 30-Aug-2020 12:22 by rmk:") (* ;; "Wrapper to call COMPARESOURCES on the Lisp source files of CDENTRY") (CL:WHEN [AND (EQ 'SOURCE (FETCH TYPE OF (FETCH INFO1 OF CDENTRY))) (EQ 'SOURCE (FETCH TYPE OF (FETCH INFO2 OF CDENTRY] (COMPARESOURCES (FETCH FULLNAME OF (FETCH INFO1 OF CDENTRY)) (FETCH FULLNAME OF (FETCH INFO2 OF CDENTRY)) EXAMINE DW? LISTSTREAM))]) ) (FILESLOAD COMPARESOURCES) (PUTPROPS COMPAREDIRECTORIES COPYRIGHT ("Venue & Xerox Corporation" 1985 1986 1987 1988 1990 1994 1998 2018 2020 2021)) (DECLARE%: DONTCOPY (FILEMAP (NIL (1632 17400 (COMPAREDIRECTORIES 1642 . 10869) (CDFILES 10871 . 15446) ( COMPAREDIRECTORIES.INFOS 15448 . 16963) (MATCHNAME 16965 . 17398)) (17401 24586 (CDPRINT 17411 . 22211 ) (CDPRINT.LINE 22213 . 24584)) (24587 26339 (CDMAP 24597 . 25293) (CDENTRY 25295 . 25463) (CDSUBSET 25465 . 26337)) (26340 31871 (BINCOMP 26350 . 30639) (EOLTYPE 30641 . 31869)) (32084 45291 ( FIND-UNCOMPILED-FILES 32094 . 35737) (FIND-UNSOURCED-FILES 35739 . 38548) (FIND-SOURCE-FILES 38550 . 40254) (FIND-COMPILED-FILES 40256 . 42334) (FIND-UNLOADED-FILES 42336 . 43080) (FIND-LOADED-FILES 43082 . 43636) (FIND-MULTICOMPILED-FILES 43638 . 45289)) (45292 53494 (CREATED-AS 45302 . 50099) ( SOURCE-FOR-COMPILED-P 50101 . 52799) (COMPILE-SOURCE-DATE-DIFF 52801 . 53492)) (53495 62474 ( FIX-DIRECTORY-DATES 53505 . 55473) (FIX-EQUIV-DATES 55475 . 56735) (COPY-COMPARED-FILES 56737 . 58861) (COPY-MISSING-FILES 58863 . 60702) (COMPILED-ON-SAME-SOURCE 60704 . 62472)) (62629 63240 ( COMPARE-ENTRY-SOURCE-FILES 62639 . 63238))))) STOP \ No newline at end of file +(DEFINE-FILE-INFO PACKAGE "INTERLISP" READTABLE "INTERLISP" BASE 10) (FILECREATED "21-Feb-2021 20:37:49"  {DSK}kaplan>Local>medley3.5>git-medley>lispusers>COMPAREDIRECTORIES.;282 65535 changes to%: (FNS EOLTYPE EOLTYPE.SHOW) (VARS COMPAREDIRECTORIESCOMS) previous date%: "21-Feb-2021 00:14:38" {DSK}kaplan>Local>medley3.5>git-medley>lispusers>COMPAREDIRECTORIES.;278) (* ; " Copyright (c) 1985-1988, 1990, 1994, 1998, 2018, 2020-2021 by Venue & Xerox Corporation. ") (PRETTYCOMPRINT COMPAREDIRECTORIESCOMS) (RPAQQ COMPAREDIRECTORIESCOMS ( (* ;; "Compare the contents of two directories.") (FNS COMPAREDIRECTORIES CDFILES COMPAREDIRECTORIES.INFOS MATCHNAME MEDLEY-FIX-DIRS) (VARS MEDLEY-FIX-DIRS) (FNS CDPRINT CDPRINT.LINE) (FNS CDMAP CDENTRY CDSUBSET) (FNS BINCOMP EOLTYPE EOLTYPE.SHOW) (RECORDS CDENTRY CDINFO) (* ;; "look for compiled files older than the sources") (FNS FIND-UNCOMPILED-FILES FIND-UNSOURCED-FILES FIND-SOURCE-FILES FIND-COMPILED-FILES FIND-UNLOADED-FILES FIND-LOADED-FILES FIND-MULTICOMPILED-FILES) (FNS CREATED-AS SOURCE-FOR-COMPILED-P COMPILE-SOURCE-DATE-DIFF) (FNS FIX-DIRECTORY-DATES FIX-EQUIV-DATES COPY-COMPARED-FILES COPY-MISSING-FILES COMPILED-ON-SAME-SOURCE) [VARS (ONESECOND (IDIFFERENCE (IDATE "1-Jan-2020 12:00:01") (IDATE "1-Jan-2020 12:00:00"] (INITVARS (LASTCDENTRIES NIL)) (COMS (FNS COMPARE-ENTRY-SOURCE-FILES) (FILES COMPARESOURCES)))) (* ;; "Compare the contents of two directories.") (DEFINEQ (COMPAREDIRECTORIES [LAMBDA (DIR1 DIR2 SELECT FILEPATTERNS EXTENSIONSTOAVOID USEDIRECTORYDATE OUTPUTFILE ALLVERSIONS) (* ; "Edited 7-Jan-2021 23:21 by rmk:") (* ;; "Compare the contents of two directories, e.g., for change-control purposes. Compares files matching FILEPATTERN (or *.*;) on DIR1 and DIR2, listing which is newer, or when one is not found on the other. If SELECT is or contains SAME/=, BEFORE/<, AFTER/>, then files where DIR1 is the same as, earlier than, or later than DIR2 are selected. SELECT= NIL is the same as (< >), T is the same as (< > =). Also allows selection based on file-length criteria.") (* ;; "") (* ;; "Unless USEDIRECTORYDATE, comparison is with respect to the the LISP filecreated dates if evailable.") (* ;; "") (* ;; "If OUTPUTFILE is NIL, the list of compared entries is returned. Otherwise the selected entries are printed on OUTPUTFILE (T for the display).") [SETQ SELECT (SELECTQ SELECT (NIL '(< > -* *-)) (T '(< > -* *- =)) (FOR S IN (MKLIST SELECT) COLLECT (SELECTQ S ((AFTER >) '>) ((BEFORE <) '<) ((SAME SAMEDATE =) '=) (AUTHOR 'AUTHOR) (-* '-*) (*- '*-) (~= '~=) (ERROR "UNRECOGNIZED SELECT PARAMETER" S] (PROG (INFOS1 INFOS2 CANDIDATES SELECTED COMPAREDATE DEPTH1 DEPTH2) [SETQ COMPAREDATE (INTERSECTION SELECT '(< > =] (* ;; "DIRECTORYNAME here to get unrelativized specifications for header.") (* ;; "Allow all subdirectories if a directory ends in *, but peel it off for the resolution") (CL:WHEN (EQ '* (NTHCHAR DIR1 -1)) (SETQ DEPTH1 T) (SETQ DIR1 (SUBSTRING DIR1 1 -2))) (CL:WHEN (EQ '* (NTHCHAR DIR2 -1)) (SETQ DEPTH2 T) (SETQ DIR2 (SUBSTRING DIR2 1 -2))) (SETQ DIR1 (OR (DIRECTORYNAME (OR DIR1 T)) (ERROR "DIRECTORY DOES NOT EXIST" DIR1))) (SETQ DIR2 (OR (DIRECTORYNAME (OR DIR2 T)) (ERROR "DIRECTORY DOES NOT EXIST" DIR2))) (PRINTOUT T "Comparing " DIR1 6 "vs. " DIR2 T "as of " (DATE) " selecting " SELECT " ... ") (SETQ INFOS1 (COMPAREDIRECTORIES.INFOS (CDFILES DIR1 FILEPATTERNS EXTENSIONSTOAVOID ALLVERSIONS DEPTH1) USEDIRECTORYDATE)) (SETQ INFOS2 (COMPAREDIRECTORIES.INFOS (CDFILES DIR2 FILEPATTERNS EXTENSIONSTOAVOID ALLVERSIONS DEPTH2) USEDIRECTORYDATE)) (CL:UNLESS (AND INFOS2 INFOS1) (RETURN)) (* ;; "At this point the CAR of each info is the atomic match-name. Peel it off to produce candidate entries.") (* ;;  "Look through all of the I2's because multiple versions (if VERSIONS) have the same matchname") [SETQ CANDIDATES (FOR I1 IN INFOS1 JOIN (IF ALLVERSIONS THEN (OR (FOR I2 IN INFOS2 WHEN (EQ (CAR I2) (CAR I1)) COLLECT (LIST (CAR I1) (CDR I1) (CDR I2))) (CONS (LIST (CAR I1) (CDR I1) NIL))) ELSE (CONS (LIST (CAR I1) (CDR I1) (CDR (ASSOC (CAR I1) INFOS2] (* ;; "Could be some 2's without 1's") (SORT [NCONC CANDIDATES (FOR I2 IN INFOS2 UNLESS (ASSOC (CAR I2) CANDIDATES) COLLECT (LIST (CAR I2) NIL (CDR I2] T) (* ;; "CANDIDATES is now a sorted list of the form (matchname entry1 entry2) where an entry consists of (fullname date length author)") (* ;; "Do the SELECT filtering and insert the date relation.") [SETQ SELECTED (FOR C MATCHNAME INFO1 INFO2 IDATE1 IDATE2 DATEREL BINCOMP IN CANDIDATES EACHTIME (SETQ MATCHNAME (POP C)) (SETQ INFO1 (POP C)) (SETQ INFO2 (POP C)) (IF (AND INFO1 INFO2) THEN (SETQ IDATE1 (IDATE (FETCH DATE OF INFO1))) (SETQ IDATE2 (IDATE (FETCH DATE OF INFO2))) (SETQ DATEREL (IF (IGREATERP IDATE1 IDATE2) THEN '> ELSEIF (ILESSP IDATE1 IDATE2) THEN '< ELSE '=)) ELSE (* ;; "Just for printing--no comparison") (SETQ DATEREL '*)) WHEN (IF (AND INFO1 INFO2) THEN (CL:WHEN (OR (NULL COMPAREDATE) (SELECTQ DATEREL (> (MEMB '> SELECT)) (< (MEMB '< SELECT)) (= (MEMB '= SELECT)) (SHOULDNT))) (SETQ BINCOMP (BINCOMP (FETCH FULLNAME OF INFO1) (FETCH FULLNAME OF INFO2) T (FETCH EOL OF INFO1) (FETCH EOL OF INFO2))) (* ;; "We want the ~= test to reflect exact byte equivalence, including the same EOL. We use the BINCOMP value below to indicate EOL differences, so we check it here.") [NOT (AND (MEMB '~= SELECT) BINCOMP (EQ (FETCH EOL OF INFO1) (FETCH EOL OF INFO2]) ELSEIF INFO1 THEN (* ;; "OK if INFO2 is missing?") (MEMB '*- SELECT) ELSE (* ;; "OK if INFO1 is missing?") (MEMB '-* SELECT)) COLLECT (CREATE CDENTRY MATCHNAME _ MATCHNAME INFO1 _ INFO1 DATEREL _ DATEREL INFO2 _ INFO2 EQUIV _ (CL:UNLESS (EQ DATEREL '*) BINCOMP] (PRINTOUT T (LENGTH SELECTED) " entries" T) (PUSH SELECTED (LIST DIR1 DIR2 SELECT (DATE))) (SETQ LASTCDENTRIES SELECTED) (CL:UNLESS OUTPUTFILE (RETURN SELECTED)) (RETURN (CDPRINT SELECTED OUTPUTFILE (MEMB 'AUTHOR SELECT SELECT]) (CDFILES [LAMBDA (DIR FILEPATTERNS EXTENSIONSTOAVOID ALLVERSIONS DEPTH) (* ; "Edited 16-Oct-2020 13:42 by rmk:") (* ;; "Returns a list of fullnames for files that satisfy the criteria") (* ;; "For each name returned by (DIRECTORY DIR), assumes that FILEPATTERNS applies to the suffix after the directory (i.e. after NAMEPOS). That includes possibly subdirectories, dotted files in ultimate file names, and versions.") (* ;; " Exclude subdirectories unless FILEPATTERNS includes *>*") (* ;; " Exclude dotted files (.xxx) unless FILEPATTERNS includes .*") (* ;; " Exclude files with extensions in EXTENSIONSTOAVOID (*=NIL does no filtering)") (* ;; " Exclude older versions unless ALLVERSIONS=T") (* ;; " DEPTH is the number of subdirectories below the ones specified in DIR (NIL top-level of DIR only, T = any depth)") (* ;; "Resolve relative directories, so we can suppress subdirectory matches. ") (SETQ EXTENSIONSTOAVOID (MKLIST (U-CASE EXTENSIONSTOAVOID))) [SETQ FILEPATTERNS (MKLIST (OR FILEPATTERNS '*] (FOR FP FN FPNAME FPEXT EXCLUDEDOTTED (TOPDIR _ (DIRECTORYNAME (OR DIR T))) IN FILEPATTERNS JOIN [SETQ FPNAME (U-CASE (FILENAMEFIELD FP 'NAME] [SETQ FPEXT (U-CASE (FILENAMEFIELD FP 'EXTENSION] (CL:UNLESS FPNAME (IF FPEXT THEN (* ;; ".XY") (SETQ FPNAME (PACK* "." FPEXT)) ELSE (SETQ FPNAME '*))) (CL:UNLESS FPEXT (SETQ FPEXT '*)) (SETQ EXCLUDEDOTTED (NEQ (CHARCODE %.) (CHCON1 FPNAME))) (SETQ FN (PACKFILENAME.STRING 'VERSION (CL:IF ALLVERSIONS '* "") 'DIRECTORY TOPDIR 'NAME '* 'EXTENSION '*)) (* ;; "DEPTH is the number of internal %">%"") [IF (EQ DEPTH T) THEN (SETQ DEPTH MAX.SMALLP) ELSEIF DEPTH ELSE (SETQ DEPTH (BIND (CNT _ 0) (POS _ 0) (FNDIR _ (FILENAMEFIELD FN 'DIRECTORY)) WHILE (SETQ POS (STRPOS ">" FNDIR (ADD1 POS))) DO (ADD CNT 1) FINALLY (RETURN CNT] (FOR FULLNAME NAME EXT THISDEPTH IN (DIRECTORY FN) EACHTIME [SETQ NAME (U-CASE (FILENAMEFIELD FULLNAME 'NAME] [SETQ EXT (U-CASE (FILENAMEFIELD FULLNAME 'EXTENSION] (CL:UNLESS NAME (IF EXT THEN (* ;; ".XY") (SETQ NAME (PACK* "." EXT)) (SETQ EXT NIL))) (CL:WHEN (AND EXCLUDEDOTTED (EQ (CHARCODE %.) (CHCON1 NAME))) (GO $$ITERATE)) (SETQ THISDEPTH (BIND (CNT _ 0) (POS _ 0) (FNDIR _ (FILENAMEFIELD FULLNAME 'DIRECTORY)) WHILE (SETQ POS (STRPOS ">" FNDIR (ADD1 POS))) DO (ADD CNT 1) FINALLY (RETURN CNT))) (* ;; "An empty subdirectory may appear without name or extensions") WHEN (AND (OR NAME EXT) (OR (EQ FPNAME '*) (EQ FPNAME NAME)) (OR (EQ FPEXT '*) (EQ FPEXT EXT))) UNLESS [OR (IGREATERP THISDEPTH DEPTH) (AND EXT (OR (MEMB '* EXTENSIONSTOAVOID) (MEMB EXT EXTENSIONSTOAVOID] COLLECT FULLNAME) FINALLY (CL:UNLESS $$VAL (PRINTOUT T "No relevant files in " TOPDIR T]) (COMPAREDIRECTORIES.INFOS [LAMBDA (FILES USEDIRECTORYDATE) (* ; "Edited 13-Oct-2020 08:42 by rmk:") (* ;; "Value is a list of CDINFOS with the match-name consed on to the front") (FOR FULLNAME TYPE LDATE IN FILES COLLECT (* ;; "GDATE/IDATE in case Y2K") (SETQ LDATE (FILEDATE FULLNAME)) (* ; "Is it a Lisp file?") (CONS (MATCHNAME FULLNAME) (CREATE CDINFO FULLNAME _ FULLNAME DATE _ [GDATE (IDATE (IF USEDIRECTORYDATE THEN (GETFILEINFO FULLNAME 'CREATIONDATE) ELSEIF (OR LDATE (GETFILEINFO FULLNAME 'CREATIONDATE] LENGTH _ (GETFILEINFO FULLNAME 'LENGTH) AUTHOR _ (GETFILEINFO FULLNAME 'AUTHOR) TYPE _ (IF LDATE THEN (CL:IF (MEMB (FILENAMEFIELD FULLNAME 'EXTENSION) *COMPILED-EXTENSIONS*) 'COMPILED 'SOURCE) ELSE (PRINTFILETYPE FULLNAME)) EOL _ (EOLTYPE FULLNAME]) (MATCHNAME [LAMBDA (NAME) (* ; "Edited 5-Sep-2020 13:41 by rmk:") (* ;; "The NAME.DIR for matching related files") (LET ((M (PACKFILENAME 'HOST NIL 'VERSION NIL 'DIRECTORY NIL 'BODY NAME))) (* ;; "Strip off the nuisance period") (CL:IF (EQ (CHARCODE %.) (NTHCHARCODE M -1)) (SUBATOM M 1 -2) M)]) (MEDLEY-FIX-DIRS [LAMBDA (DIRS) (* ; "Edited 8-Jan-2021 23:00 by rmk:") (* ;  "Edited 4-Jan-2021 15:42 by larry") (for X in (OR (MKLIST DIRS) MEDLEY-FIX-DIRS) join (FIX-DIRECTORY-DATES (MEDLEYDIR (PRINT X T]) ) (RPAQQ MEDLEY-FIX-DIRS ("sources" "library" "lispusers" "internal/library" "greetfiles" "docs>Documentation Tools" "cltl2" "clos")) (DEFINEQ (CDPRINT [LAMBDA (CDENTRIES FILE PRINTAUTHOR) (* ; "Edited 13-Oct-2020 08:38 by rmk:") (* ;; "Typically CDENTRIES will have a header. If not, we fake one up, at least for the directories and today's date.") (CL:UNLESS CDENTRIES (PRINTOUT T T "Note: Using LASTCDENTRIES" T T) (SETQ CDENTRIES LASTCDENTRIES)) (RESETLST (LET (INFO1 TEXT STREAM DATE1POS ENDDATE1 DIR1 DIR2 (HEADER (CAR CDENTRIES)) NCHARSDIR1) (CL:UNLESS (STRINGP (CADR HEADER)) (SETQ HEADER (LIST [FOR E IN CDENTRIES WHEN (FETCH INFO1 OF E) DO (RETURN (PACKFILENAME 'NAME NIL 'EXTENSION NIL 'VERSION NIL 'BODY (FETCH FULLNAME OF (FETCH INFO1 OF E] [FOR E IN CDENTRIES WHEN (FETCH INFO2 OF E) DO (RETURN (PACKFILENAME 'NAME NIL 'EXTENSION NIL 'VERSION NIL 'BODY (FETCH FULLNAME OF (FETCH INFO2 OF E] NIL (DATE))) (PUSH CDENTRIES HEADER)) (SETQ DIR1 (CAR HEADER)) (SETQ NCHARSDIR1 (NCHARS DIR1)) (SETQ DIR2 (CADR HEADER)) (CL:UNLESS (SETQ STREAM (GETSTREAM FILE 'OUTPUT T)) [RESETSAVE (SETQ STREAM (OPENSTREAM (PACKFILENAME 'EXTENSION 'TXT 'BODY FILE) 'OUTPUT 'NEW)) '(PROGN (CLOSEF? OLDVALUE]) (CL:WHEN DIR1 (PRINTOUT STREAM "Comparing " DIR1 6 "vs. " DIR2 T "as of " (CADDDR HEADER)) (CL:WHEN (CADDR HEADER) (PRINTOUT STREAM " selecting " (CADDR HEADER))) (PRINTOUT STREAM -2 (LENGTH (CDR CDENTRIES)) " entries" T T)) (LINELENGTH 1000 STREAM) (* ; "Don't wrap") (* ;; "DATE1POS is the position of the first character of INFO1's date, used for tabbing. We have to measure the filename, date, size, and author if desired") (IF (CDR CDENTRIES) THEN (FOR E INFO1 (MAXDATE1WIDTH _ 0) (SPACEWIDTH _ 1) (PARENWIDTH _ 2) IN (CDR CDENTRIES) WHEN (SETQ INFO1 (FETCH INFO1 OF E)) LARGEST [SETQ MAXDATE1WIDTH (IMAX MAXDATE1WIDTH (NCHARS (FETCH DATE OF INFO1] (IPLUS (- (NCHARS (FETCH FULLNAME OF INFO1)) NCHARSDIR1) (NCHARS (FETCH LENGTH OF INFO1)) (CL:IF PRINTAUTHOR (IPLUS SPACEWIDTH PARENWIDTH (NCHARS (FETCH AUTHOR OF INFO1))) 0)) FINALLY (* ;;  "First 4 for width of equiv. $$EXTREME is NIL if there are no INFO1's") (SETQ DATE1POS (IPLUS (OR $$EXTREME 10) 4 (ITIMES 3 SPACEWIDTH))) (SETQ ENDDATE1 (IPLUS DATE1POS MAXDATE1WIDTH) )) (FOR E IN (CDR CDENTRIES) DO (CDPRINT.LINE STREAM E PRINTAUTHOR DATE1POS ENDDATE1 NCHARSDIR1 (NCHARS DIR2))) ELSE (PRINTOUT T "CDENTRIES is empty" T)) (AND STREAM (CLOSEF? STREAM))))]) (CDPRINT.LINE [LAMBDA (STREAM ENTRY PRINTAUTHOR DATE1POS ENDDATE1 NCHARSDIR1 NCHARSDIR2) (* ; "Edited 9-Jan-2021 10:12 by rmk:") (* ;; "Format one line of the directory comparison listing. If PRINTAUTHOR and AUTHOR1 or AUTHOR2 are non-NIL, list the author in parens; otherwise omit it.") (LET ((INFO1 (FETCH INFO1 OF ENTRY)) (INFO2 (FETCH INFO2 OF ENTRY))) (PRINTOUT STREAM (SELECTQ (FETCH EQUIV OF ENTRY) (T "==") (NIL " ") (CONCAT (SELECTQ (CAR (FETCH EQUIV OF ENTRY)) (CR 'C) (LF 'L) (CRLF 2) "x") (SELECTQ (CADR (FETCH EQUIV OF ENTRY)) (CR 'C) (LF 'L) (CRLF 2) "x"))) " ") (CL:WHEN INFO1 (PRINTOUT STREAM (SUBSTRING (FETCH FULLNAME OF INFO1) (ADD1 NCHARSDIR1) NIL (CONSTANT (CONCAT))) " ") (CL:WHEN PRINTAUTHOR (PRINTOUT STREAM "(" (FETCH AUTHOR OF INFO1) ") ")) (PRINTOUT STREAM (FETCH LENGTH OF INFO1) .TAB0 DATE1POS (FETCH DATE OF INFO1))) (PRINTOUT STREAM .TAB0 ENDDATE1 " " (FETCH DATEREL OF ENTRY) " ") (CL:WHEN INFO2 (PRINTOUT STREAM (FETCH DATE OF INFO2) " " (SUBSTRING (FETCH FULLNAME OF INFO2) (ADD1 NCHARSDIR2) NIL (CONSTANT (CONCAT))) " ") (CL:WHEN PRINTAUTHOR (PRINTOUT STREAM "(" (FETCH AUTHOR OF INFO2) ") ")) (PRINTOUT STREAM (FETCH LENGTH OF INFO2))) (TERPRI STREAM]) ) (DEFINEQ (CDMAP [LAMBDA (CDENTRIES FN) (* ; "Edited 6-Sep-2020 15:58 by rmk:") (CL:UNLESS CDENTRIES (PRINTOUT T T "Note: Using LASTCDENTRIES" T T) (SETQ CDENTRIES LASTCDENTRIES)) (FOR CDE MATCHNAME INFO1 DATEREL INFO2 EQUIV IN (CDR CDENTRIES) DECLARE (SPECVARS MATCHNAME INFO1 DATEREL INFO2 EQUIV) EACHTIME (SETQ MATCHNAME (FETCH MATCHNAME OF CDE)) (SETQ INFO1 (FETCH INFO1 OF CDE)) (SETQ DATEREL (FETCH DATEREL OF CDE)) (SETQ INFO2 (FETCH INFO2 OF CDE)) (SETQ EQUIV (FETCH EQUIV OF CDE)) DO (APPLY* FN CDE]) (CDENTRY [LAMBDA (MATCHNAME CDENTRIES) (* ; "Edited 5-Sep-2020 21:09 by rmk:") (ASSOC MATCHNAME (OR CDENTRIES LASTCDENTRIES]) (CDSUBSET [LAMBDA (CDENTRIES FN) (* ; "Edited 15-Sep-2020 13:49 by rmk:") (CL:UNLESS CDENTRIES (PRINTOUT T T "Note: Using LASTCDENTRIES" T T) (SETQ CDENTRIES LASTCDENTRIES)) (CONS (CAR CDENTRIES) (FOR CDE MATCHNAME INFO1 DATEREL INFO2 EQUIV IN (CDR CDENTRIES) DECLARE (SPECVARS MATCHNAME INFO1 DATEREL INFO2 EQUIV) EACHTIME (SETQ MATCHNAME (FETCH MATCHNAME OF CDE)) (SETQ INFO1 (FETCH INFO1 OF CDE)) (SETQ DATEREL (FETCH DATEREL OF CDE)) (SETQ INFO2 (FETCH INFO2 OF CDE)) (SETQ EQUIV (FETCH EQUIV OF CDE)) WHEN (APPLY* FN CDE) COLLECT CDE]) ) (DEFINEQ (BINCOMP [LAMBDA (FILE1 FILE2 EOLDIFFOK EOL1 EOL2) (* ; "Edited 13-Oct-2020 08:53 by rmk:") (* ;; "Returns T if FILE1 and FILE2 are byte-equivalent. Returns EOLDIFF if they are byte equivalent except for CR/LF/CRLF exchanges. ") (* ;; "If EOLDIFFOK, return indicates that the files are the same except for EOL mappings. If EOL1 and EOL2 are not provided, they are computed here.") (IF (IEQP (GETFILEINFO FILE1 'LENGTH) (GETFILEINFO FILE2 'LENGTH)) THEN [CL:WITH-OPEN-FILE (STREAM1 FILE1 :DIRECTION :INPUT) (CL:WITH-OPEN-FILE (STREAM2 FILE2 :DIRECTION :INPUT) (SETFILEINFO STREAM1 'ENDOFSTREAMOP (FUNCTION NILL)) (* ;; "Simpler code to recompute eol's even if provided") (BIND B1 B2 EOL1 EOL2 EOLDIFF WHILE (SETQ B1 (\BIN STREAM1)) UNLESS (EQ B1 (SETQ B2 (\BIN STREAM2))) DO (CL:UNLESS (AND EOLDIFFOK (SELCHARQ B1 (CR (CL:WHEN (EQ EOL1 'LF) (RETURN NIL)) (SETQ EOL1 'CR) (SETQ EOL2 'LF) (EQ B2 (CHARCODE LF))) (LF (CL:WHEN (EQ EOL1 'CR) (RETURN NIL)) (SETQ EOL1 'LF) (SETQ EOL2 'CR) (EQ B2 (CHARCODE CR))) NIL)) (RETURN NIL)) (CL:UNLESS EOLDIFF (SETQ EOLDIFF (LIST EOL1 EOL2))) FINALLY (RETURN (OR EOLDIFF T] ELSEIF EOLDIFFOK THEN (* ;; "Lengths are different possibly because of CRLF to CR/LF substitutions.") (* ;;  "More complex code could detect the EOLTYPE incrementally without separate passes, but ...") (CL:UNLESS EOL1 (SETQ EOL1 (EOLTYPE FILE1))) (CL:UNLESS EOL2 (SETQ EOL2 (EOLTYPE FILE2))) (CL:WHEN (IF [AND (EQ EOL1 'CRLF) (MEMB EOL2 '(LF CR] ELSEIF [AND (EQ EOL2 'CRLF) (MEMB EOL1 '(LF CR] THEN (SWAP FILE1 FILE2)) (* ;; "FILE1 is now CRLF, FILE2 is not. If FILE1 isn't longer, it can't have a CRLF that corresponds to a CR or LF.") (CL:WHEN (IGREATERP (GETFILEINFO FILE1 'LENGTH) (GETFILEINFO FILE2 'LENGTH)) [CL:WITH-OPEN-FILE (STREAM1 FILE1 :DIRECTION :INPUT) (CL:WITH-OPEN-FILE (STREAM2 FILE2 :DIRECTION :INPUT) (SETFILEINFO STREAM1 'ENDOFSTREAMOP (FUNCTION NILL)) (BIND B1 B2 EOLDIFF WHILE (SETQ B1 (\BIN STREAM1)) UNLESS (EQ B1 (SETQ B2 (\BIN STREAM2))) DO (CL:UNLESS [AND (EQ (CHARCODE CR) B1) (EQ (CHARCODE LF) (\BIN STREAM1)) (MEMB B2 (CHARCODE (CR LF] (RETURN NIL)) (CL:UNLESS EOLDIFF (SETQ EOLDIFF (LIST EOL1 EOL2))) FINALLY (RETURN (OR EOLDIFF T]))]) (EOLTYPE [LAMBDA (FILE SHOWCONTEXT) (* ; "Edited 21-Feb-2021 20:34 by rmk:") (* ;; "Returns the EOLCONVENTION of FILE if it only sees one kind, NIL if it can't decide.") (* ;; "If SHOWCONTEXT, it is the number of bytes before and after an EOL inconsistency (e.g. seeing CR after having seen LF) that will be displayed on the TTY. The position of the inconsistency will be marked with ##.") (SELECTQ SHOWCONTEXT (NIL) (T (SETQ SHOWCONTEXT 100)) (CL:UNLESS (FIXP SHOWCONTEXT) (ERROR "SHOWCONTEXT must be an integer" SHOWCONTEXT))) (CL:WITH-OPEN-FILE (STREAM FILE :DIRECTION :INPUT) (SETFILEINFO STREAM 'ENDOFSTREAMOP (FUNCTION NILL)) (BIND EOLTYPE DO (SELCHARQ (OR (\BIN STREAM) (RETURN EOLTYPE)) (CR (IF (EQ (CHARCODE LF) (\PEEKBIN STREAM T)) THEN (\BIN STREAM) (IF (MEMB EOLTYPE '(LF CR)) THEN (CL:UNLESS (EOLTYPE.SHOW SHOWCONTEXT EOLTYPE 'LF STREAM) (RETURN NIL)) ELSE (SETQ EOLTYPE 'CRLF)) ELSEIF (MEMB EOLTYPE '(LF CRLF)) THEN (CL:UNLESS (EOLTYPE.SHOW SHOWCONTEXT EOLTYPE 'CR STREAM) (RETURN NIL)) ELSE (SETQ EOLTYPE 'CR))) (LF (IF (MEMB EOLTYPE '(CR CRLF)) THEN (CL:UNLESS (EOLTYPE.SHOW SHOWCONTEXT EOLTYPE 'LF STREAM) (RETURN NIL)) ELSE (SETQ EOLTYPE 'LF))) NIL]) (EOLTYPE.SHOW [LAMBDA (SHOWCONTEXT OLDTYPE NEWTYPE STREAM) (* ; "Edited 21-Feb-2021 20:20 by rmk:") (* ;; "Returns T if we should continue") (CL:WHEN SHOWCONTEXT (LET ((FILEPOS (GETFILEPTR STREAM))) (COPYBYTES STREAM T (IDIFFERENCE FILEPOS SHOWCONTEXT) FILEPOS) (PRINTOUT T OLDTYPE "->" NEWTYPE " " FILEPOS T) (COPYBYTES STREAM T FILEPOS (IPLUS FILEPOS SHOWCONTEXT)) (TERPRI T) (CL:WHEN (EQ 'Y (ASKUSER NIL NIL "Continue? ")) (PRINTOUT T T "-------" T T) (SETFILEPTR STREAM FILEPOS) T)))]) ) (DECLARE%: EVAL@COMPILE (RECORD CDENTRY (MATCHNAME INFO1 DATEREL INFO2 . EQUIV)) (RECORD CDINFO (FULLNAME DATE LENGTH AUTHOR TYPE EOL)) ) (* ;; "look for compiled files older than the sources") (DEFINEQ (FIND-UNCOMPILED-FILES [LAMBDA (FILES DFASLMARGIN COMPILEEXTS) (* ; "Edited 20-Sep-2020 23:04 by rmk:") (* ; "Edited 3-Nov-94 15:17 by jds") (* ;; "Produces a list of the source files in FILES that have no corresponding compiled file") (* ;; "This determines whether there is at least one compiled file. If there are two or more, that's a problem") (* ;; "We want the most recent version only") (* ;; "Source files have a 2-element created-as with a non-NIL date") (SETQ FILES (FOR F IN (OR (LISTP FILES) (FILDIR FILES)) UNLESS (MEMB (SETQ F (PACKFILENAME 'VERSION NIL 'BODY F)) $$VAL) COLLECT F)) (FOR F SCREATION FILES IN FILES WHEN (AND (CADR (SETQ SCREATION (CREATED-AS F))) (NOT (CDDR SCREATION))) WHEN [SETQ FILES (FOR CEXT CF IN (OR COMPILEEXTS *COMPILED-EXTENSIONS*) WHEN (SETQ CF (INFILEP (PACKFILENAME 'EXTENSION CEXT 'VERSION NIL 'BODY F))) COLLECT (CL:WHEN (SOURCE-FOR-COMPILED-P SCREATION CF DFASLMARGIN) (RETURN NIL)) CF FINALLY (* ;; "If we found some compiled files, they weren't on this source. If there weren't any compiled files to check, maybe there weren't any functions.") (* ;;  "NLSETQ because we don't want to stop if there is an error, typically from a package problem") (RETURN (OR $$VAL (LET [(FCOMS (CAR (NLSETQ (GETDEF (FILECOMS F) 'VARS F] (IF (NULL FCOMS) THEN (* ;;  "GETDEF caused an error. Maybe a package problem. ") (AND NIL 'NOCOMMANDS) ELSEIF (INFILECOMS? NIL '(FUNCTIONS FNS) FCOMS) THEN T] COLLECT (CONS F (SELECTQ FILES (T NIL) (NOCOMMANDS (CONS "No commands")) (FOR CF IN FILES COLLECT (* ;;  "Positive means that compiled is later than source, normal order but maybe by too much.") (* ;;  "Negative means that compiled came before source. Odd") (LIST CF (COMPILE-SOURCE-DATE-DIFF CF SCREATION]) (FIND-UNSOURCED-FILES [LAMBDA (FILES DFASLMARGIN COMPILEEXTS) (* ; "Edited 15-Sep-2020 15:32 by rmk:") (* ; "Edited 3-Nov-94 15:17 by jds") (* ;;  "Produces a list of compiled FILES for which no source file can be found in the same directory.") (* ;; "The source date in at least one DFASL was off by a second, maybe some sort of IDATE rounding? So, give a margin.") (* ;; "We want the most recent version only. Check CREATED-AS to make sure it really is a compiled file.") (* ;; "Sort to get lcoms and dfasls next to each other.") (LET (CCREATEDS) (SETQ CCREATEDS (FOR CEXT FOUND CCREATED INSIDE (OR COMPILEEXTS *COMPILED-EXTENSIONS*) JOIN (FOR CF IN [OR (LISTP FILES) (FILDIR (PACKFILENAME 'EXTENSION CEXT 'VERSION "" 'BODY '*] WHEN (CDDR (SETQ CCREATED (CREATED-AS CF))) UNLESS (MEMBER CCREATED $$VAL) COLLECT CCREATED))) (* ;; "CCREATEDS is now a list of CREATED-AS items") (FOR CC SF IN CCREATEDS UNLESS (AND [SETQ SF (INFILEP (PACKFILENAME 'EXTENSION NIL 'VERSION NIL 'BODY (CAR CC] (SOURCE-FOR-COMPILED-P (SETQ SF (CREATED-AS SF)) CC DFASLMARGIN)) COLLECT [LIST (CAR CC) (AND SF (LIST (CAR SF) (ROUND (COMPILE-SOURCE-DATE-DIFF CC SF] FINALLY (RETURN (SORT $$VAL (FUNCTION (LAMBDA (CF1 CF2) (ALPHORDER (FILENAMEFIELD (CAR CF1) 'NAME) (FILENAMEFIELD (CAR CF2) 'NAME]) (FIND-SOURCE-FILES [LAMBDA (CFILES SDIRS DFASLMARGIN) (* ; "Edited 9-Sep-2020 12:26 by rmk:") (* ;; "Returns (CFILE . SFILES) pairs where CFILE is a Lisp compiled file in CFILES SFILES is a list of source files in SDIRS that CFILE was compiled on.") (* ;; "This suggests that one of CFILES should be copied to the SFILE directory.") (SETQ SDIRS (FOR SD INSIDE (OR SDIRS T) COLLECT (DIRECTORYNAME SD))) (SORT (FOR CF SFILES CNAME CCREATED IN (OR (LISTP CFILES) (FILDIR CFILES)) WHEN (AND (SETQ CNAME (INFILEP CF)) (CDDR (SETQ CCREATED (CREATED-AS CF))) (SETQ SFILES (FOR SD SF IN SDIRS WHEN (AND (SETQ SF (INFILEP (PACKFILENAME 'NAME (FILENAMEFIELD CF 'NAME) 'BODY SD))) (SOURCE-FOR-COMPILED-P SF CCREATED DFASLMARGIN)) COLLECT SF))) COLLECT (CONS CNAME SFILES)) (FUNCTION (LAMBDA (P1 P2) (ALPHORDER (FILENAMEFIELD (CAR P1)) (FILENAMEFIELD (CAR P2]) (FIND-COMPILED-FILES [LAMBDA (SFILES CDIRS DFASLMARGIN) (* ; "Edited 9-Sep-2020 12:26 by rmk:") (* ;; "Returns (SFILE . CFILES) pairs where SFILE is a Lisp source file in SFILES CFILES is a list of compiled files in CDIRS that were compiled on SFILE.") (* ;; "FILEDATE is true for source files and compiled files") (* ;; "This suggests that one of CFILES should be copied to the SFILE directory.") (SETQ CDIRS (FOR CD INSIDE (OR CDIRS T) COLLECT (DIRECTORYNAME CD))) (SORT (FOR SF CFILES SNAME SCREATED IN (OR (LISTP SFILES) (FILDIR SFILES)) WHEN [AND (SETQ SNAME (INFILEP SF)) (SETQ SCREATED (CREATED-AS SF)) (NOT (CDDR SCREATED)) (SETQ CFILES (FOR CEXT (ROOT _ (FILENAMEFIELD SNAME 'NAME)) IN *COMPILED-EXTENSIONS* JOIN (FOR CD CF IN CDIRS WHEN (AND (SETQ CF (INFILEP (PACKFILENAME 'NAME ROOT 'EXTENSION CEXT 'BODY CD))) (SOURCE-FOR-COMPILED-P SCREATED CF DFASLMARGIN)) COLLECT CF] COLLECT (CONS SNAME CFILES )) (FUNCTION (LAMBDA (P1 P2) (ALPHORDER (FILENAMEFIELD (CAR P1)) (FILENAMEFIELD (CAR P2]) (FIND-UNLOADED-FILES [LAMBDA (FILES) (* ; "Edited 9-Sep-2020 19:35 by rmk:") (* ;; "Returns the files in FILES that don't have FILECREATED properties and presumably are therefore not loaded in the current sysout.") (FOR F IN (OR (LISTP FILES) (FILDIR FILES)) WHEN (AND (SETQ F (INFILEP (CL:IF (LISTP F) (CAR F) F))) (FILEDATE F)) UNLESS (GETP (FILENAMEFIELD F 'NAME) 'FILEDATES) COLLECT F]) (FIND-LOADED-FILES [LAMBDA (ROOTFILENAMES) (* ; "Edited 19-Sep-2020 07:20 by rmk:") (FOR RN INSIDE ROOTFILENAMES WHEN (GETP RN 'FILEDATES) COLLECT (CONS RN (FOR F IN LOADEDFILELST WHEN (EQ RN (FILENAMEFIELD F 'NAME)) COLLECT F]) (FIND-MULTICOMPILED-FILES [LAMBDA (FILES SHOWINFO) (* ; "Edited 20-Sep-2020 20:57 by rmk:") (* ;; "Returns a list of names for files in FILES that have multiple compilations") (LET (SFILES) (FOR F EXT NAME IN (OR (LISTP FILES) (FILDIR FILES)) WHEN (MEMB (SETQ EXT (FILENAMEFIELD F 'EXTENSION)) *COMPILED-EXTENSIONS*) DO (SETQ NAME (FILENAMEFIELD F 'NAME)) (* ;; "PUSHNEW because we haven't filtered out versions") (PUSHNEW [CDR (OR (ASSOC NAME SFILES) (CAR (PUSH SFILES (CONS NAME] EXT)) (FOR S IN SFILES WHEN (CDDR S) COLLECT (IF SHOWINFO THEN `[,(CAR S) ,(CADAR (FIND-LOADED-FILES (CAR S))) ,(CREATED-AS (CAR S)) ,@(FOR EXT IN (SORT (CDR S)) COLLECT (CREATED-AS (PACKFILENAME 'EXTENSION EXT 'BODY (CAR S] ELSE (CAR S]) ) (DEFINEQ (CREATED-AS [LAMBDA (FILE) (* ; "Edited 20-Sep-2020 23:06 by rmk:") (* ;; "For lisp source files, returns (filecreatename filecreateddate)") (* ;; "For lisp compiled files, returns (cfilename cfiledate sfilecreatename sfilecreateddate)") (* ;; "For other files, (fullfilename NIL)") (* ;; "The cfilename is just the current directory name for DFASLs.") (* ;; "So: (CADR value) is non-NIL for Lisp files. Of those, (CDDR value) is non-NIL for compiled files.") (* ;; "We disable the package delimiter because the atoms in changes may have a packages that we don't know.") (CL:WITH-OPEN-FILE (STREAM FILE :DIRECTION :INPUT) (LET (FILEDATE FILENAME SOURCEDATE SOURCENAME LINE POS) [IF (EQ (CHARCODE %() (SKIPSEPRCODES STREAM)) THEN (* ; "Managed source or LCOM") (RESETLST [LET (FORM SFORM (RDTBL (FIND-READTABLE "OLD-INTERLISP-FILE"))) (SETQ POS (GETFILEPTR STREAM)) (READCCODE STREAM) (IF (EQ 'DEFINE-FILE-INFO (RATOM STREAM RDTBL)) THEN (* ;; "Reading is package-safe") (SETFILEPTR STREAM POS) (SETQ FORM (READ STREAM RDTBL)) (SETQ RDTBL (FIND-READTABLE (LISTGET (CDR FORM) :READTABLE))) ELSE (SETFILEPTR STREAM POS)) (CL:WHEN (EQ 'PACKAGEDELIM (GETSYNTAX '%: RDTBL)) [RESETSAVE (SETSYNTAX '%: 'OTHER RDTBL) `(SETSYNTAX %: PACKAGEDELIM ,RDTBL]) (* ;; "One way or the other, we're ready for the filecreated") (CL:WHEN (EQ (CHARCODE %() (SKIPSEPRCODES STREAM)) (SETQ FORM (READ STREAM RDTBL)) (CL:WHEN (MEMB (U-CASE (CAR FORM)) '(FILECREATED IL%:FILECREATED)) (* ;; "IL%%:FILECREATED because we screwed the readtable.") (IF [STREQUAL "compiled on " (CAR (LISTP (CADDR FORM] THEN (* ; "LCOM, get source info") (IF [AND (EQ (CHARCODE %() (SKIPSEPRCODES STREAM)) (MEMB [U-CASE (CAR (SETQ SFORM (READ STREAM RDTBL] '(FILECREATED IL%:FILECREATED] THEN (SETQ FILENAME (FULLNAME STREAM)) (SETQ FILEDATE (CADR FORM)) (SETQ SOURCENAME (CADDR SFORM)) (SETQ SOURCEDATE (CADR SFORM)) ELSE (SETQ FILENAME (FULLNAME STREAM)) (SETQ FILEDATE (CADR FORM))) ELSE (SETQ FILENAME (CADDR FORM)) (SETQ FILEDATE (CADR FORM)))))]) ELSEIF (SETQ POS (STRPOS "XCL Compiler output for source file " (SETQ LINE (CL:READ-LINE STREAM)) 1 NIL NIL T)) THEN (* ; "DFASL compiled?") (SETQ SOURCENAME (SUBATOM LINE POS)) (CL:WHEN (SETQ POS (STRPOS "Source file created " (SETQ LINE (CL:READ-LINE STREAM)) 1 NIL NIL T)) [SETQ SOURCEDATE (GDATE (IDATE (SUBSTRING LINE POS] (CL:WHEN (SETQ POS (STRPOS "FASL file created " (SETQ LINE (CL:READ-LINE STREAM)) 1 NIL NIL T)) [SETQ FILEDATE (GDATE (IDATE (SUBSTRING LINE POS]))] (* ;; "Revert filenames to Interlisp package if needed:") (CL:WHEN (STRPOS "IL:" FILENAME) (SETQ FILENAME (SUBATOM FILENAME 4))) (CL:WHEN (STRPOS "IL:" SOURCENAME) (SETQ SOURCENAME (MKATOM SOURCENAME 4))) (* ;; "Return DATE NIL if file is not a Lisp file") `(,(OR FILENAME (FULLNAME STREAM)) ,(AND FILEDATE (GDATE (IDATE FILEDATE))) ,@(CL:WHEN SOURCENAME (LIST SOURCENAME (GDATE (IDATE SOURCEDATE))))]) (SOURCE-FOR-COMPILED-P [LAMBDA (SOURCE COMPILED DFASLMARGIN) (* ; "Edited 31-Oct-2020 09:12 by rmk:") (* ;; "There seems to be some variation between the source dates in dfasl files and the filecreated date in the sources, they often don't match exactly. But if they are within DFASLMARGIN, we assume a match. We require exact date match for LCOMS") (* ;; "This is needed for dfasl files created before they recorded the source filecreated name and date instead of the directory source name and date when compile took place.") (* ;; "") (* ;; "DFASLMARGIN is a pair (after before) where we assume a match if the compiled date is no more than after minutes after the source date and no more than before minuts before (the diff is negative then).") (* ;; "A single positive integer x is interpreted as (x 0). A single negative integer x is interpreted as (-x x) (before or after x).") (* ;; "Default is (20 0).") (* ;; "T is positive or negative infinity") (CL:UNLESS (LISTP SOURCE) (SETQ SOURCE (CREATED-AS SOURCE))) (CL:UNLESS (LISTP COMPILED) (SETQ COMPILED (CREATED-AS COMPILED))) (SETQ DFASLMARGIN (IF (NULL DFASLMARGIN) THEN (* ;;  "If compiled is later than source by less than 20 minutes, it's probably OK") '(20 0) ELSEIF (EQ T DFASLMARGIN) THEN '(T 0) ELSEIF (LISTP DFASLMARGIN) ELSEIF (NOT (FIXP DFASLMARGIN)) THEN (ERROR "ILLEGAL DFASLMARGIN" DFASLMARGIN) ELSEIF (MINUSP DFASLMARGIN) THEN (LIST (MINUS DFASLMARGIN) DFASLMARGIN) ELSE (LIST DFASLMARGIN 0))) (OR (EQUAL (CAR SOURCE) (CADDR COMPILED)) (EQUAL (CADR SOURCE) (CADDDR COMPILED)) (AND [EQ 'DFASL (U-CASE (FILENAMEFIELD (CAR COMPILED) 'EXTENSION] (LET ((TIMEDIFF (COMPILE-SOURCE-DATE-DIFF COMPILED SOURCE))) (* ;; "If compiled was no more than 20 minutes later, it's probably OK. Of no more than DFASLMARGIN earlier, if it is negative.") (AND (OR (EQ T (CAR DFASLMARGIN)) (LEQ TIMEDIFF (CAR DFASLMARGIN))) (OR (EQ T (CADR DFASLMARGIN)) (GEQ TIMEDIFF (CADR DFASLMARGIN]) (COMPILE-SOURCE-DATE-DIFF [LAMBDA (CFILE SFILE) (* ; "Edited 20-Sep-2020 22:59 by rmk:") (* ;; "Positive means that compiled is later than source, normal order but maybe by too much. Negative means that compiled came before source, i.e., compiled on a source that didn't yet exist.") (* ;; "Value is in minutes") (ROUND (FQUOTIENT [IDIFFERENCE [IDATE (CADDDR (OR (LISTP CFILE) (CREATED-AS CFILE] (IDATE (CADR (OR (LISTP SFILE) (CREATED-AS SFILE] (TIMES 60 ONESECOND]) ) (DEFINEQ (FIX-DIRECTORY-DATES [LAMBDA (FILES MARGIN) (* ; "Edited 30-Oct-2020 22:01 by rmk:") (* ;; "For Lisp source and compiled files, ensures that the directory file date corresponds to the filecreated date. Returns the list of files whose dates were changed.") (* ;; "This allows for the fact that directory dates that are no later than, say, 30 seconds of the filecreated date are probably OK--the directory date may be set when the file is closed.") (* ;; "Use IDATEs in case FDCDATE is not Y2K.") (* ;; "Stop if directory date is more than 2 minutes earlier than the filecreated date. Earlier could be because the dates are asserted at different points in the filing process. But 2 minutes is worth thinking about. Returning from HELP will get them aligned.") (SETQ MARGIN (ITIMES (OR MARGIN 2) 60 ONESECOND)) (FOR F DIDATE FCDATE IN (OR (LISTP FILES) (FILDIR FILES)) WHEN (SETQ FCDATE (FILEDATE F)) UNLESS (IEQP (SETQ DIDATE (GETFILEINFO F 'ICREATIONDATE)) (SETQ FCDATE (IDATE FCDATE))) COLLECT (CL:WHEN (IGREATERP (IDIFFERENCE FCDATE DIDATE) MARGIN) (HELP "DIRECTORY DATE EARLIER THAN FILECREATED DATE" (LIST F (GDATE DIDATE) (GDATE FCDATE)))) (SETFILEINFO F 'ICREATIONDATE FCDATE) F]) (FIX-EQUIV-DATES [LAMBDA (CDENTRIES) (* ; "Edited 1-Sep-2020 16:21 by rmk:") (* ;; "For every entry whose files are EQUIVALENT and whose filedates are different, sets the directory of the file with the later date to be the date of the one with the earlier date. This preumes that the later one must have been a copy. ") (CL:UNLESS CDENTRIES (PRINTOUT T "Note: Using LASTCDENTRIES" T) (SETQ CDENTRIES LASTCDENTRIES)) (FOR CDE EARLY LATE IN (CDR CDENTRIES) WHEN (FETCH EQUIV OF CDE) UNLESS (EQ '= (FETCH DATEREL OF CDE)) COLLECT (SELECTQ (FETCH DATEREL OF CDE) (> (SETQ EARLY (FETCH INFO2 OF CDE)) (SETQ LATE (FETCH INFO1 OF CDE))) (< (SETQ EARLY (FETCH INFO1 OF CDE)) (SETQ LATE (FETCH INFO2 OF CDE))) (SHOULDNT)) (SETFILEINFO (FETCH FULLNAME OF LATE) 'ICREATIONDATE (GETFILEINFO (FETCH FULLNAME OF EARLY) 'ICREATIONDATE)) (FETCH FULLNAME OF LATE]) (COPY-COMPARED-FILES [LAMBDA (CDENTRIES TARGET MATCHNAMES) (* ; "Edited 1-Sep-2020 16:20 by rmk:") (* ;; "Copies source files to target files whose matchname belongs to MATCHNAMES, if given.") (* ;; "TARGET is 1 or 2, indicating which side of the CD entry is the target. Value is the list of matchnames whose files have been copied.") (* ;; "Directory filedates and other properties are preserved.") (CL:UNLESS (MEMB TARGET '(1 2)) (ERROR "INVALID TARGET" TARGET)) (CL:UNLESS CDENTRIES (PRINTOUT T "Note: Using LASTCDENTRIES" T) (SETQ CDENTRIES LASTCDENTRIES)) (SETQ MATCHNAMES (MKLIST MATCHNAMES)) (FOR CDE SINFO TINFO MATCHNAME IN (CDR CDENTRIES) EACHTIME (SETQ SINFO (FETCH INFO1 OF CDE)) (SETQ TINFO (FETCH INFO2 OF CDE)) (CL:WHEN (EQ TARGET 1) (SWAP SINFO TINFO)) (SETQ MATCHNAME (FETCH MATCHNAME OF CDE)) WHEN (AND (FETCH FULLNAME OF SINFO) (FETCH FULLNAME OF TINFO)) UNLESS (AND MATCHNAMES (NOT (MEMB MATCHNAME MATCHNAMES))) COLLECT (COPYFILE (FETCH FULLNAME OF SINFO) (PACKFILENAME 'VERSION NIL 'BODY (FETCH FULLNAME OF TINFO))) MATCHNAME]) (COPY-MISSING-FILES [LAMBDA (CDENTRIES TARGET MATCHNAMES) (* ; "Edited 1-Sep-2020 16:21 by rmk:") (* ;; "Copies source files to target files whose matchname belongs to MATCHNAMES, if given.") (* ;; "TARGET is 1 or 2, indicating which side of the CD entry is the target. Value is the list of matchnames whose files have been copied.") (* ;; "Directory filedates and other properties are preserved.") (CL:UNLESS (MEMB TARGET '(1 2)) (ERROR "INVALID TARGET" TARGET)) (CL:UNLESS CDENTRIES (PRINTOUT T "Note: Using LASTCDENTRIES" T) (SETQ CDENTRIES LASTCDENTRIES)) (CL:UNLESS (STRINGP (CADR (CAR CDENTRIES))) (ERROR "(CAR CDENTRIES) IS NOT A VALID PARAMETER LIST" (CAR CDENTRIES))) (SETQ MATCHNAMES (MKLIST MATCHNAMES)) (FOR CDE SINFO TINFO TDIR MATCHNAME (TDIR _ (CL:IF (EQ TARGET 1) (CAAR CDENTRIES) (CADAR CDENTRIES))) IN (CDR CDENTRIES) EACHTIME (SETQ MATCHNAME (FETCH MATCHNAME OF CDE)) (SETQ SINFO (FETCH INFO1 OF CDE)) (SETQ TINFO (FETCH INFO2 OF CDE)) (CL:WHEN (EQ TARGET 1) (SWAP SINFO TINFO)) WHEN (AND (FETCH FULLNAME OF SINFO) (NOT (FETCH FULLNAME OF TINFO))) UNLESS (AND MATCHNAMES (NOT (MEMB MATCHNAME MATCHNAMES))) COLLECT (* ;; "Using the source fullname in the target should preserve the version number") (COPYFILE (FETCH FULLNAME OF SINFO) (PACKFILENAME 'BODY TDIR 'BODY (FETCH FULLNAME OF SINFO))) MATCHNAME]) (COMPILED-ON-SAME-SOURCE [LAMBDA (CDENTRIES) (* ; "Edited 9-Sep-2020 13:00 by rmk:") (* ;; "Returms a subset of CDENTRIES consisting of files that are compiled on the same source (i.e. their source names or dates are the same). Preserves the header.") (CDSUBSET CDENTRIES (FUNCTION (LAMBDA (CDE) (DECLARE (USEDFREE INFO1 INFO2)) (LET (CREATED1 CREATED2) (CL:WHEN [AND (EQ 'COMPILED (FETCH TYPE OF INFO1)) (EQ 'COMPILED (FETCH TYPE OF INFO2)) [CDDR (SETQ CREATED1 (CREATED-AS (FETCH FULLNAME OF INFO1] (CDDR (SETQ CREATED2 (CREATED-AS (FETCH FULLNAME OF INFO2] (OR (EQUAL (CADDR CREATED1) (CADDR CREATED2)) (EQUAL (CADDDR CREATED1) (CADDDR CREATED2))))]) ) (RPAQ ONESECOND (IDIFFERENCE (IDATE "1-Jan-2020 12:00:01") (IDATE "1-Jan-2020 12:00:00"))) (RPAQ? LASTCDENTRIES NIL) (DEFINEQ (COMPARE-ENTRY-SOURCE-FILES [LAMBDA (CDENTRY LISTSTREAM EXAMINE DW?) (* ; "Edited 30-Aug-2020 12:22 by rmk:") (* ;; "Wrapper to call COMPARESOURCES on the Lisp source files of CDENTRY") (CL:WHEN [AND (EQ 'SOURCE (FETCH TYPE OF (FETCH INFO1 OF CDENTRY))) (EQ 'SOURCE (FETCH TYPE OF (FETCH INFO2 OF CDENTRY] (COMPARESOURCES (FETCH FULLNAME OF (FETCH INFO1 OF CDENTRY)) (FETCH FULLNAME OF (FETCH INFO2 OF CDENTRY)) EXAMINE DW? LISTSTREAM))]) ) (FILESLOAD COMPARESOURCES) (PUTPROPS COMPAREDIRECTORIES COPYRIGHT ("Venue & Xerox Corporation" 1985 1986 1987 1988 1990 1994 1998 2018 2020 2021)) (DECLARE%: DONTCOPY (FILEMAP (NIL (1712 17919 (COMPAREDIRECTORIES 1722 . 10949) (CDFILES 10951 . 15526) ( COMPAREDIRECTORIES.INFOS 15528 . 17043) (MATCHNAME 17045 . 17478) (MEDLEY-FIX-DIRS 17480 . 17917)) ( 18092 25299 (CDPRINT 18102 . 22902) (CDPRINT.LINE 22904 . 25297)) (25300 27052 (CDMAP 25310 . 26006) ( CDENTRY 26008 . 26176) (CDSUBSET 26178 . 27050)) (27053 33994 (BINCOMP 27063 . 31352) (EOLTYPE 31354 . 33319) (EOLTYPE.SHOW 33321 . 33992)) (34207 47414 (FIND-UNCOMPILED-FILES 34217 . 37860) ( FIND-UNSOURCED-FILES 37862 . 40671) (FIND-SOURCE-FILES 40673 . 42377) (FIND-COMPILED-FILES 42379 . 44457) (FIND-UNLOADED-FILES 44459 . 45203) (FIND-LOADED-FILES 45205 . 45759) (FIND-MULTICOMPILED-FILES 45761 . 47412)) (47415 55617 (CREATED-AS 47425 . 52222) (SOURCE-FOR-COMPILED-P 52224 . 54922) ( COMPILE-SOURCE-DATE-DIFF 54924 . 55615)) (55618 64597 (FIX-DIRECTORY-DATES 55628 . 57596) ( FIX-EQUIV-DATES 57598 . 58858) (COPY-COMPARED-FILES 58860 . 60984) (COPY-MISSING-FILES 60986 . 62825) (COMPILED-ON-SAME-SOURCE 62827 . 64595)) (64752 65363 (COMPARE-ENTRY-SOURCE-FILES 64762 . 65361))))) STOP \ No newline at end of file diff --git a/lispusers/COMPAREDIRECTORIES.LCOM b/lispusers/COMPAREDIRECTORIES.LCOM index 7a669cbf6265ddda6d75faddcc3714b0c444f27f..7acbaa7d67c3c8e1b23cd8a0a0cfede10869a9d6 100644 GIT binary patch delta 2443 zcmZWrO>7%g5cb;rfizZ|Hfd>_(&;u$?Shki`{!L(lsNV}-qx|5_2$P24oSQurcRvN z4r!jL0v=Qq5X*9Z~XlR_tBT!8&ka^iKLvlh$x5zL?I?cW3v9{OKR6bxe-}%N)_h@HT@9P#4rST z&?E7c8afLC5G^K)-V(v$wdM>7OfQ6vTC7iX6joH{hBm`5m> z$`-tAJULWI=cdCDnu=QmdS>d_`7o^1oR`bh)w-J-?t9=^L5LAKCTWizd&aMY>9wie z)-C@pJYStQYR?_5{Y^qg3?)uI*U%fe)4z|FS@f!l)Y0$$vagPMX(JE69tiMn+4BI zbfFWJL^bykMD@Igvh5`+kjP~daV%1ZNCG6RObUo7Hz5OrsK9;+8YJ+W86g^NyF>{( zU_i7h7y@Lb#~2{#QQ*VCgA|p351~{v{Do2`z@s22m|fJ+5$$4BbnPO>LAxlS8sI7B zA-GE(dr@XOTqsO~QWk>1U+00vxZe8v^p${LQCq3ADuN^Qn`LT0HFb$m0CI&4GQ~q4vJJ|K0TAlWOq% z_*BNkmLa1407KyWvR)k@1x;v6MTc=(j_`em4;KdR`|9|<7HAr!kuayzFCxb|nf@3V z@~iFA=|jC&RO|qx00g-aA4fJIu)7|N9T@3hIV``==Z}}xPn61A?`&2|bN<&Q_{1$j z+^xbUy*i{eqj+W}mdNA^21A*51%y(Ed9l0*byH_B6-O8lBR{sVdk<7v^X62-3d3BR_hIzb6|EE%F7LB!Koo}caC=H)q_I& zV)odUxchN%_f9fi(DqzS!0QIaDwIdkn{QWjNA*na@Y56!hOUZ&X*{6rAK~B zyN?^Euk9U_H`8s@Y?fW0L^)j=x$oNBip7#!!2=8q?(1*eyFjn)Jwo5#x3B4KO1t$f zA3{_pNJ)mxo+NKu*iE{}Bt+gO6@s&-&4PrnV4G=(phLWHWwKycfQ{bKA`>QZ`Qm7i zQ80q3qecS208Qjk)eD7h(qUFlv(2@lx+@3~VQf1=MokFSB6}$YVm{LTAP)#g42kM; zqrAN8Jj)mYA?%V3^SXp@mNMxN!$(^E!+pp0U`LXcl6oTGuCS}e+Kb#!h2BUXY#qCN zr~AUA)61QskE(<8?$jW?KQ%b)!OWYx2&|lyEI=@kG_tl;ghbNJrgAuuFTll%Kw*0P zC?=LoMkcd`+@zH-k~mZ|SUdXabcPGk@1{@C-=~c}baF@S10#+mXD0Rs*=&ap{kCY+ zt(opa!4R8@?ek0)sJm!3^YMv^hx3LIU$3s#=AAm0sZyS+&DL(XtNO}n-KjAx1AJ|H zwlch9UZ^=v<3;=jg!Uj`s?OIhC9CtROU`m*wo$GwgI%pw*xva{qk?~#P*&?4-^v$n GasLC2FlEsI delta 1322 zcmZux&2Jk;6!$s-rKLvsq$N$7hZWoFNOpH-XFu!~B*$Ji3;u}L+lYK2f@3wQOl`#h zA@z_8e?Y4d2M{Vk2!wM#W^D%}SZVjY`Tc(H&FuVU z_nRlu@k8m($|M+xx3_L3s6q))HBCvHDR*z`BaPj5tJd7B^RFZrHnce0eRh5_^ub=e z*8H%w+o*3t{gc*SJCf#=GZS2rW*pG|>6A<+!?3-zkFrpBkRVE$v9xA};~_b37p$yf zWn3!<;s09)Z%WaBE##u1fvz$_p|-Pi>qfl|`^`uiY~;m;BD`oV@v6Gl19O^`)S= zr0pAGcyhlNzwHh?HveswdC^NT>5`(uvSXJBs3{%X%p$0!(MJrOAI!}p)l-0KeTt-@ zdVR6#V(B$d71G0LO5soEW)8GdKuz^2RUIgyq(3iJSWiJx8n~8&%K>8cX=0qxy4n|F zd$Clao~me1g*FkzJbC&1m2qrKr1JzI<97ktWrbAf9 zs1h7uBn-k~pe6>vQyqoKQ>OVmrKZmlgaPm-VEQR$njtVuRDob5W(b4#>KpG21)1ip z&+8IKek8xik0Y{3|BC2A#`yd3+x&jCCDJ4LExsAM8Du7Z8v9d(Be^buFEvp5YmgZT zrXrBenIU-@RLCd_L*sw~x@2XF=m;)Mpv?PW0fr7lir-1#Qz;mbD*nmvAFqfb(-|8L zRN&!DONZ`uBXsiMYNxV$RuGM|Cl98V{8Q$SJ+%TIzwYtj@gRP&2mUB39t|+}cE-^NIjiJ4s~GK4 zzKp~IT)6@v{`*>y6LSg+D(eM)Z+#lG$LrQq-hMZcv$5JOJFAHt{$Hx79^a^puVkDm n$mNn%#nz=9$RQqk&*o1y&W*~2Ox5kqWc&IT8@BiT=56V3*gs8m