source: branches/working-0711/ccl/level-1/l1-pathnames.lisp @ 9788

Last change on this file since 9788 was 9788, checked in by gb, 11 years ago

If they don't care, why should we ?

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 31.0 KB
Line 
1;;;-*-Mode: LISP; Package: CCL -*-
2;;;
3;;;   Copyright (C) 1994-2001 Digitool, Inc
4;;;   Portions copyright (c) 2001 Clozure Associates.
5;;;   This file is part of OpenMCL. 
6;;;
7;;;   OpenMCL is licensed under the terms of the Lisp Lesser GNU Public
8;;;   License , known as the LLGPL and distributed with OpenMCL as the
9;;;   file "LICENSE".  The LLGPL consists of a preamble and the LGPL,
10;;;   which is distributed with OpenMCL as the file "LGPL".  Where these
11;;;   conflict, the preamble takes precedence. 
12;;;
13;;;   OpenMCL is referenced in the preamble as the "LIBRARY."
14;;;
15;;;   The LLGPL is also available online at
16;;;   http://opensource.franz.com/preamble.html
17
18
19
20;; L1-pathnames.lisp
21;-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_
22;ANSI CL logical pathnames
23
24(in-package "CCL")
25
26(defun heap-image-name ()
27  (let* ((p (%null-ptr)))
28    (declare (dynamic-extent p))
29    (%get-cstring (%get-kernel-global-ptr 'image-name p))))
30
31(defloadvar *heap-image-name* (heap-image-name))
32
33(defloadvar *command-line-argument-list*
34  (let* ((argv (%null-ptr))
35         (res ()))
36    (declare (dynamic-extent argv))
37    (%get-kernel-global-ptr 'argv argv)
38    (do* ((i 0 (+ i target::node-size))
39          (arg (%get-ptr argv i) (%get-ptr argv i)))
40         ((%null-ptr-p arg) (nreverse res))
41      (declare (fixnum i))
42      (push (%get-cstring arg) res))))
43
44;These are used by make-pathname
45(defun %verify-logical-component (name type)
46  (when (and name (neq name :unspecific))
47    (setq name (ensure-simple-string name))
48    (when (or (eql 0 (length name))
49              (%str-member *pathname-escape-character* name) ;; Hmm, why?
50              (%path-mem "/;" name))
51      (error "Illegal logical pathname ~A component ~S" type name)))
52  name)
53
54
55(defun verify-logical-host-name (host)
56  (or (and host
57           (%verify-logical-component host "host")
58           (%str-assoc host %logical-host-translations%)
59           host)
60      (host-error host)))
61
62(defun %logical-version-component (version)
63  (if (or (fixnump version)
64          (stringp version)
65          (memq version '(nil :wild :newest :unspecific)))
66    version
67    (require-type version '(or fixnum string (member nil :wild :newest :unspecific)))))
68
69(defun logical-pathname-translations (host)
70  "Return the (logical) host object argument's list of translations."
71  (setq host (verify-logical-host-name host))
72  (let ((translations (%str-assoc host %logical-host-translations%)))
73    (unless translations (host-error host))
74    (%cdr translations)))
75
76(defun logical-host-p (host)
77  (%str-assoc host %logical-host-translations%))
78
79(defun host-error (host) ; supposed to be a type-error
80  (signal-type-error host  '(satisfies logical-host-p) "~S is not a defined logical host"))
81
82(defun set-logical-pathname-translations (host list)
83  (setq host (%verify-logical-component  host "host"))
84  (let ((old (%str-assoc host %logical-host-translations%))
85        (new (let ((%logical-host-translations% (cons (list host) %logical-host-translations%)))
86               ;; Do this in the context when host is defined, so no errors.
87               (mapcar #'(lambda (trans)
88                           (destructuring-bind (from to &rest ignored) trans
89                             (declare (ignore ignored))
90                             (let ((from-path (parse-namestring from host))
91                                   (to-path (pathname to)))
92                               (list (require-type from-path 'logical-pathname) to-path))))
93                       list))))
94    (if old
95      (progn (%rplaca old host) (%rplacd old new))
96      (push (cons host new) %logical-host-translations%)))
97  list)
98
99(defsetf logical-pathname-translations set-logical-pathname-translations)
100
101;;; doesnt check if already there - adds at front
102(defun add-logical-pathname-translation (host translation)
103  (let ((trans (%str-assoc host  %logical-host-translations%)))
104    (if (not trans)
105      (set-logical-pathname-translations host (list translation))
106      (let ((new (destructuring-bind (from to &rest ignored) translation
107                   (declare (ignore ignored))
108                   (list (parse-namestring from host) (pathname to)))))
109        (rplacd trans (cons new (cdr trans)))
110        (cdr trans)))))
111
112(defun %component-match-p (name wild) 
113  (if (or (eq name :unspecific)(eq name :wild)(eq name :wild-inferiors)(and (stringp name) (or  (string= name "*")(string= name "**"))))
114    (setq name nil)) 
115  (if (or (eq wild :unspecific)(eq wild :wild)(eq wild :wild-inferiors)(and (stringp wild) (or (string= wild "*")(string= wild "**"))))
116    (setq wild nil))
117  (cond ((null name) 
118         (null wild))
119        ((null wild)
120         t)
121        ((not (and (stringp name) (stringp wild)))
122         (eq name wild))
123        (t (%path-str*= name wild))))
124
125(defun translate-directory (source from to reversible &optional thost)
126  (declare (ignore thost)) ;; leftover from a mac kludge.
127  (let* ((result (translate-directory2 (cdr source)(cdr from)(cdr to) reversible))
128         (relative-p (eq (car source) :relative)))
129    (cond ((and (not relative-p)(eq result (cdr source))) (or source (list :absolute)))
130          ((and (not relative-p)(eq result (cdr to))) to)
131          (t (cons (car (or to source from)) result)))))
132
133
134
135(defun translate-directory2 (source from to reversible)
136  ; we already know it matches
137  (let (result srest match tfirst trest twild)
138    (multiple-value-setq (tfirst trest twild)
139                         (%split-ccdirectory to))
140    (when (and to (not twild))
141      (return-from translate-directory2 to))
142    (multiple-value-bind (ffirst frest fwild)
143                         (%split-ccdirectory from)
144      (setq srest (nthcdr (length ffirst) source))
145      (cond ((eq fwild '**)
146             (setq match (nth-value 1 (%pathname-match-dir1 srest frest t)))               
147             (cond ((eq twild '**)
148                    (setq result (nconc tfirst match))
149                    (setq srest (nthcdr (length match) srest)))
150                   (t (return-from translate-directory2
151                        (translate-directory2 source (nconc ffirst match frest)
152                                              to reversible)))))
153            ((eq twild '**)
154             (let ((length (length tfirst)))
155               (setq srest (nthcdr length source))
156               (setq frest (nthcdr length from))
157               (setq  match (nth-value 1 (%pathname-match-dir1 srest trest t)))
158               (cond ((null  match)
159                      (setq result tfirst))
160                     (t (setq srest (nthcdr (setq length (length match)) srest))
161                        (setq frest (nthcdr length frest))
162                        (setq result (nconc tfirst match))))))
163            (t
164             (cond ((null fwild)
165                    ; to has a wild component e.g. *abc, from is not wild
166                    ; by defintion source is also not wild
167                    ; which random source component gets plugged in here??
168                    (setq srest (nthcdr (length tfirst) source))
169                    (setq frest (nthcdr (length tfirst) source))))
170             (let ((part (translate-component
171                                (car srest) (car frest)(car trest) reversible)))
172               (if (null part)(setq result tfirst)
173                   (progn
174                     (setq part (list part))
175                     (setq result (nconc tfirst part)))))
176             (setq srest (cdr srest) frest (cdr frest) trest (cdr trest))))
177      (when trest 
178        (let ((foo (translate-directory2 srest frest trest reversible)))
179          (when foo (setq result (nconc result foo))))))
180    result))
181
182; cc stands for cdr canonical
183; ("abc" "**" "def" => ("abc") ("def")
184; ("abc" "*de") => ("abc") ("*de")
185(defun %split-ccdirectory (dir)
186  (let ((pos 0) (wildp nil)(rest dir))
187    (dolist (e dir)
188      (case e
189        (:wild (setq wildp '*))
190        (:wild-inferiors 
191         (setq wildp '**)
192         (setq rest (cdr rest)))
193        (:up nil)
194        (t 
195         (when (%path-mem "*" e)
196           (cond ((string= e "**")
197                  (setq rest (cdr rest))
198                  (setq wildp '**))
199                 ((eql 1 (length (the string e)))
200                  (setq wildp '*))
201                 (t (setq wildp t))))))
202      (when wildp (return))
203      (setq rest (cdr rest))
204      (setq pos (%i+ 1 pos)))
205    (cond ((not wildp)
206           (values dir))
207          (t (let (first)
208               (when rest (setq rest (copy-list rest)))
209               (dotimes (i pos)
210                 (declare (fixnum i))
211                 (push (car dir) first)
212                 (setq dir (cdr dir)))
213               (values (nreverse first) rest wildp))))))
214
215; could avoid calling component-match-p by checking here maybe
216; if "gazonk" "gaz*" "h*" => "honk"
217; then "gazonk" "gaz*" "*" => "onk" or is it "gazonk" (per pg 625)
218; I believe in symbolics land "gazonk" is a regular translation
219; and "onk" is a reversible translation (achieved by not doing pg 625) AHH
220; similarly is "a:" "a:**:" "**"  Nil or "a:"
221(defun translate-component (source from to &optional reversible)                   
222  (let ((orig-to to))
223    (cond 
224     ((and (consp source)(consp from)) ; source and from both logical
225      (setq source (cadr source) from (cadr from)))
226     ((or (consp source)(consp from)) ;  or neither
227      #-bccl (error "Something non-kosher in translate pathname")
228      ))
229    (when (memq from '(:wild :wild-inferiors)) (setq from "*"))
230    (when (memq source '(:wild :wild-inferiors))(setq source "*"))
231    (when (memq to '(:wild :wild-inferiors))(setq to "*"))
232    (cond ((consp to)(setq to (cadr to))))  ;??
233    (cond ((and (stringp to)(not (%path-mem "*" to)))
234           to)
235          ((and (or (not reversible)(not (stringp source))) ; <<
236                (or (null to)
237                    (and (stringp to)(or (string= to "**")(string= to "*")))))
238           source)
239          ((eq to :unspecific) to)  ; here we interpret :unspecific to mean don't want it
240          ((not (stringp source)) to)
241          (t 
242           (let ((slen (length source)) srest match spos result (f2 nil) snextpos)
243             (multiple-value-bind (tfirst trest twild)
244                                  (%split-component to)
245               (cond ((and to (not twild))(return-from translate-component to)))
246               (multiple-value-bind (ffirst frest fwild)
247                                    (%split-component from)         
248                 (cond (fwild
249                        (setq spos (if ffirst (length ffirst) 0))       ; start of source hunk
250                        (if frest (setq f2 (%split-component frest)))
251                        (setq snextpos (if f2 (%path-member f2 source spos) slen))
252                        (setq match (%substr source spos snextpos))
253                        (if frest (setq srest (%substr source snextpos slen)))
254                        (setq result (if tfirst (%str-cat tfirst match) match))
255                        (when frest 
256                          (let ((foo (translate-component srest frest trest reversible)))
257                            (when foo (setq result (%str-cat result foo))))))
258                       (t  ; to is wild, from and source are not
259                        (setq result (if tfirst (%str-cat tfirst source) source))
260                        (when trest (setq result (%str-cat result trest))))))
261               (if (consp orig-to)(progn (error "shouldnt")(list :logical result)) result) ; 7/96
262               ))))))
263
264
265(defun %path-member (small big &optional (start 0))
266  (let* ((end (length big))
267         (s-end (length small))
268         (s-start 1)
269         (c1 (%schar small 0))
270         (pstart start))
271    (if (%i> s-end end)(return-from %path-member nil))
272    (when (eql c1 *pathname-escape-character*)
273      (setq c1 (%schar small 1))
274      (setq s-start 2))     
275    (while (and (progn (if (eql (%schar big pstart) *pathname-escape-character*)
276                         (setq pstart (%i+ pstart 1)))
277                       T)
278                (%i< pstart end)
279                (neq (%schar big pstart) c1))
280      (setq pstart (%i+ pstart 1)))
281    (if (neq c1 (%schar big pstart))(return-from %path-member nil))
282    (setq start (%i+ pstart 1))
283    (while (and (progn (if (eql (%schar big start) *pathname-escape-character*)
284                         (setq start (%i+ 1 start)))
285                       (if (eql (%schar small s-start) *pathname-escape-character*)
286                         (setq s-start (%i+ 1 s-start)))
287                       T)
288                (%i< start end)
289                (%i< s-start s-end)
290                (eql (%schar big start)(%schar small s-start)))
291      (setq start (%i+ start 1) s-start (%i+ s-start 1)))
292    (cond ((= (the fixnum s-start) (the fixnum s-end))
293            pstart)
294          ((%i< start end)
295            (%path-member small big (%i+ 1 pstart)))
296          (T nil))))
297
298(defun %split-component (thing &aux pos)
299  ;"ab*cd*"  ->  "ab" "cd*" 
300  (if (or (not (typep thing 'string))(null (setq pos (%path-mem "*" thing))))
301    (values thing nil nil)
302    (let* ((len (length thing)))
303      (declare (fixnum len))
304      (values (if (%izerop pos) nil (%substr thing 0 pos))
305              (cond ((eql len (%i+ pos 1)) nil)
306                    (t 
307                     (when (eq (%schar thing (+ pos 1)) #\*)
308                       (setq pos (+ pos 1)))
309                     (cond ((eql len (%i+ pos 1)) nil)
310                           (t (%substr thing (%i+ pos 1) len)))))
311              T))))
312
313(defun translate-pathname (source from-wildname to-wildname &key reversible)
314  "Use the source pathname to translate the from-wildname's wild and
315   unspecified elements into a completed to-pathname based on the to-wildname."
316  (when (not (pathnamep source)) (setq source (pathname source)))
317  (flet ((foo-error (source from)
318           (error "Source ~S and from-wildname ~S do not match" source from)))
319    (let (r-host r-device r-directory r-name r-type r-version s-host f-host t-host)
320      (setq s-host (pathname-host source))
321      (setq f-host (pathname-host from-wildname))
322      (setq t-host (pathname-host to-wildname))
323      (if (not (%host-component-match-p s-host f-host)) (foo-error source from-wildname))
324      (setq r-host (translate-component s-host f-host t-host reversible))
325      (let ((s-dir (%std-directory-component (pathname-directory source) s-host))
326            (f-dir (%std-directory-component (pathname-directory from-wildname) f-host))
327            (t-dir (%std-directory-component (pathname-directory to-wildname) t-host)))
328        (let ((match (%pathname-match-directory s-dir f-dir)))
329          (if (not match)(foo-error source from-wildname))
330          (setq r-directory  (translate-directory s-dir f-dir t-dir reversible t-host))))
331      (let ((s-name (pathname-name source))
332            (f-name (pathname-name from-wildname))
333            (t-name (pathname-name to-wildname)))
334        (if (not (%component-match-p s-name f-name))(foo-error source from-wildname))       
335        (setq r-name (translate-component s-name f-name t-name reversible)))
336      (let ((s-type (pathname-type source))
337            (f-type (pathname-type from-wildname))
338            (t-type (pathname-type to-wildname)))
339        (if (not (%component-match-p s-type f-type))(foo-error source from-wildname))
340        (setq r-type (translate-component s-type f-type t-type reversible)))
341      (let ((s-version (pathname-version source))
342            (f-version (pathname-version from-wildname))
343            (t-version (pathname-version to-wildname)))
344        (if (not (%component-match-p s-version f-version))(foo-error source from-wildname))
345        (setq r-version (translate-component s-version f-version t-version reversible))
346        ;(if (eq r-version :unspecific)(setq r-version nil))
347        )
348      (make-pathname :device r-device :host r-host :directory r-directory
349                     :name r-name :type r-type :version r-version :defaults nil)
350      )))
351
352
353
354(defvar %empty-logical-pathname% (%cons-logical-pathname nil nil nil nil nil))
355
356(defun logical-pathname-namestring-p (string)
357  (multiple-value-bind (sstr start end) (get-sstring string)
358    (let ((host (pathname-host-sstr sstr start end t)))
359      (and host (not (eq host :unspecific))))))
360
361 
362;; This extends CL in that it allows a host-less pathname, like "foo;bar;baz".
363(defun logical-pathname (thing &aux (path thing))
364  "Converts the pathspec argument to a logical-pathname and returns it."
365  (when (typep path 'stream) (setq path (%path-from-stream path)))
366  (etypecase path
367    (logical-pathname path)
368    (pathname (report-bad-arg thing 'logical-pathname))
369    (string
370     (multiple-value-bind (sstr start end) (get-sstring path)
371       ;; Prescan the host, to avoid unknown host errors.
372       (let ((host (pathname-host-sstr sstr start end t)))
373         (when (or (null host) (eq host :unspecific))
374           (report-bad-arg path '(satisfies logical-pathname-namestring-p)))
375         (let ((%logical-host-translations% (cons (list host) %logical-host-translations%)))
376           (declare (special %logical-host-translations%))
377           ;; By calling string-to-pathname with a logical pathname as default, we force
378           ;; parsing as a logical pathname.
379           (string-to-pathname sstr start end nil %empty-logical-pathname%)))))))
380
381(defun %host-component-match-p (path-host wild-host)
382  ;; Note that %component-match-p is case sensitive.  Need a
383  ;; case-insensitive version for hosts.
384  ;; In addition, host components do not support wildcards.
385  (or (eq path-host wild-host)
386      (and (stringp path-host)
387           (stringp wild-host)
388           (string-equal path-host wild-host))))
389
390(defun pathname-match-p (pathname wildname)
391  "Pathname matches the wildname template?"
392  (let ((path-host (pathname-host pathname))
393        (wild-host (pathname-host wildname)))
394    (and
395     (%host-component-match-p path-host wild-host)
396     (%component-match-p (pathname-device pathname)(pathname-device wildname))
397     (%pathname-match-directory
398      (%std-directory-component (pathname-directory pathname) path-host)
399      (%std-directory-component (pathname-directory wildname) wild-host))
400     (%component-match-p (pathname-name pathname)(pathname-name wildname))
401     (%component-match-p (pathname-type pathname)(pathname-type wildname))
402     (%component-match-p (pathname-version pathname)(pathname-version wildname)))))
403
404
405; expects canonicalized directory - how bout absolute vs. relative?
406(defun %pathname-match-directory (path wild)
407  (cond ((equal path wild) t)
408         ; Don't allow matching absolute and relative, so that can have distinct
409         ; absolute and wild translations in logical-pathname-translations for
410         ; a host, and have them match separately.
411        ((and (consp path)(consp wild)(neq (car path) (car wild)))
412         nil)  ; one absolute & one relative ??
413        ((or ;(and (null wild)
414             ;     (let ((dir (cadr path)))
415             ;       (if (stringp dir)(string= dir "**")(eq dir :wild-inferiors))))
416             (and (null (cddr wild))
417                  (let ((dir (cadr wild)))
418                    (if (stringp dir)(string= dir "**")(eq dir :wild-inferiors))))))
419        ((null path)
420         ;; Make missing dir match (:absolute) or (:relative) - is that right?
421         (null (cdr wild)))
422        ((null wild)
423         nil)
424        (t (%pathname-match-dir0 (cdr path)(cdr wild)))))
425
426; munch on tails path and wild
427(defun %pathname-match-dir0 (path wild)
428  (flet ((only-wild (dir)
429                    (when (null (cdr dir))
430                      (setq dir (car dir))
431                      (if (stringp dir)(string= dir "**")(eq dir :wild-inferiors)))))
432    (cond ((eq path wild) t)
433          ((only-wild wild)
434           t)
435          (t (let ((result t))
436               (block nil 
437                 (while (and path wild)
438                   (let ((pathstr (car path))
439                         (wildstr (car wild)))
440                     (case wildstr
441                       (:wild (setq wildstr "*"))
442                       (:wild-inferiors (setq wildstr "**")))
443                     (case pathstr
444                       (:wild (setq pathstr "*"))
445                       (:wild-inferiors (setq pathstr "**")))
446                     (when (not 
447                            (cond ((string= wildstr "**")
448                                   (setq result (%pathname-match-dir1 path (cdr wild)))
449                                   (return-from nil))
450                                  ((%path-str*= pathstr wildstr))))
451                       (setq result nil)
452                       (return-from nil))
453                     (setq wild (cdr wild) path (cdr path))))
454                 (when (and (or path wild)(not (only-wild wild)))
455                   (setq result nil)))
456               result)))))
457
458(defun %pathname-match-dir0 (path wild)
459  (flet ((only-wild (dir)
460                    (when (null (cdr dir))
461                      (setq dir (car dir))
462                      (when (consp dir) (setq dir (cadr dir)))
463                      (if (stringp dir)(string= dir "**")(eq dir :wild-inferiors)))))
464    (cond ((eq path wild) t)
465          ((only-wild wild)
466           t)
467          (t (let ((result t))
468               (block nil 
469                 (while (and path wild)
470                   (let ((pathstr (car path))
471                         (wildstr (car wild)))                     
472                     ; allow logical to match physical today
473                     ; because one of these days these logical things will disappear!
474                     (when (consp pathstr)(setq pathstr (cadr pathstr)))
475                     (when (consp wildstr)(setq wildstr (cadr wildstr)))
476                     (case wildstr
477                       (:wild (setq wildstr "*"))
478                       (:wild-inferiors (setq wildstr "**")))
479                     (case pathstr
480                       (:wild (setq pathstr "*"))
481                       (:wild-inferiors (setq pathstr "**")))
482                     (if (or (memq wildstr '(:up :back))(memq pathstr '(:up :back))) ;; ????? <<<<
483                       (when (neq pathstr wildstr)(setq result nil) (return-from nil))
484                       (when (not 
485                              (cond ((string= wildstr "**")
486                                     (setq result (%pathname-match-dir1 path (cdr wild)))
487                                     (return-from nil))
488                                    ((%path-str*= pathstr wildstr))))
489                         (setq result nil)
490                         (return-from nil)))
491                     (setq wild (cdr wild) path (cdr path))))
492                 (when (and (or path wild)(not (only-wild wild)))
493                   (setq result nil)))
494               result)))))
495
496
497
498; wild is stuff after a "**" - looking for what matches the **  in (path)
499(defun %pathname-match-dir1 (path wild &optional cons-result)
500  (let ((match nil) pathstr wildstr)
501    (cond ((null wild)
502           (values T (if cons-result (mapcar #'(lambda (e)
503                                            (if (consp e)(cadr e) e))
504                                        path))))
505          ((%pathname-match-dir0 path wild))   ; ie ** matches nothing
506          (t 
507           (prog nil
508             AGN
509               (setq pathstr (car path) wildstr (car wild))
510               (when (consp pathstr)(setq pathstr (cadr pathstr)))
511               (when (consp wildstr)(setq wildstr (cadr wildstr)))
512               (case wildstr
513                 (:wild (setq wildstr "*"))
514                 (:wild-inferiors (setq wildstr "**")))
515               (case pathstr
516                 (:wild (setq pathstr "*"))
517                 (:wild-inferiors (setq pathstr "**")))
518               (until (or (not (consp path))
519                          (%path-str*= pathstr wildstr))
520                 (when cons-result (push pathstr match))
521                 (setq path (cdr path))
522                 (setq pathstr (car path))
523                 (when (consp pathstr)(setq pathstr (cadr pathstr))))
524               ;; either got a match - w and path both have the thing we looked for
525               ;; or path is empty
526               (when (null path)(return nil))
527               (let ((path1 (cdr path))(wild (cdr wild)))
528                 (when (and (null path1)(null wild))
529                   (return (values t (when match (nreverse match)))))
530                 (cond ((%pathname-match-dir0 path1 wild)  ; is the rest happy too?
531                        (return (values t (nreverse match))))
532                       (t (when cons-result (push pathstr match)) ; nope, let ** eat more
533                          (setq path (cdr path))
534                          (go AGN)))))))))
535
536; three times bigger and 3 times slower - does it matter?
537(defun %path-str*= (string pattern)
538  (multiple-value-bind (string s-start s-end) (get-sstring string)
539    (multiple-value-bind (pattern p-start p-end) (get-sstring pattern)
540      (path-str-sub pattern string p-start s-start p-end s-end))))
541
542(defun path-str-sub (pattern str p-start s-start p-end s-end)
543  (declare (fixnum p-start s-start p-end s-end)
544           (type simple-base-string pattern str))
545  (declare (optimize (speed 3)(safety 0)))
546  (let ((p (%scharcode pattern p-start))
547        (esc (char-code *pathname-escape-character*)))
548    (cond 
549     ((eq p (char-code #\*))
550      ; starts with a * find what we looking for unless * is last in which case done
551      (loop ; lots of *'s same as one
552        (when (eq (%i+ 1 p-start)  p-end)
553          (return-from path-str-sub t))
554        (if (eq (%schar pattern (%i+ 1 p-start)) #\*)
555          (setq p-start (1+ p-start))
556          (return)))
557      (let* ((next* (%path-mem "*" pattern (%i+ 1 p-start)))
558             (len (- (or next* p-end) (%i+ 1 p-start))))
559        (loop
560          (when (> (+ s-start len) s-end)(return nil))
561          (let ((res (find-str-pattern pattern str (%i+ 1 p-start) s-start (or next* p-end) s-end))) 
562            (if (null res)
563              (return nil)
564              (if (null next*)
565                (if (eq res s-end)
566                  (return t))                 
567                (return (path-str-sub pattern str next* (+ s-start len) p-end s-end)))))
568          (setq s-start (1+ s-start)))))
569     (t (when (eq p esc)
570          (setq p-start (1+ p-start))
571          (setq p (%scharcode pattern p-start)))
572        (let* ((next* (%path-mem "*" pattern (if (eq p (char-code #\*))(%i+ 1 p-start) p-start)))
573               (this-s-end (if next* (+ s-start (- next* p-start)) s-end)))
574          (if (> this-s-end s-end)
575            nil
576            (if  (path-str-match-p pattern str p-start s-start (or next* p-end) this-s-end)
577              (if (null next*)
578                t                 
579                (path-str-sub pattern str next* this-s-end p-end s-end)))))))))
580
581; find match of pattern between start and end in str
582; rets one past end of pattern in str or nil
583(defun find-str-pattern (pattern str p-start s-start p-end s-end)
584  (declare (fixnum p-start s-start p-end s-end)
585           (type simple-base-string pattern str))
586  (declare (optimize (speed 3)(safety 0)))
587  (let* ((first-p (%scharcode pattern p-start))
588         (esc (char-code *pathname-escape-character*)))
589    (when (and (eq first-p esc) (not (eq (setq p-start (1+ p-start)) p-end)))
590      (setq first-p (%scharcode pattern p-start)))
591    (do* ((i s-start (1+ i))
592          (last-i (%i- s-end (%i- p-end p-start))))
593         ((> i last-i) nil)
594      (declare (fixnum i last-i))
595      (let ((s (%scharcode str i)))
596        (when (eq first-p s)
597          (do* ((j (1+ i) (1+ j))
598                (k (1+ p-start)(1+ k)))
599               ((>= k p-end) (return-from find-str-pattern j))
600            (declare (fixnum j k))
601            (let* ((p (%scharcode pattern k))
602                   (s (%scharcode str j)))
603              (when (and (eq p esc) (< (setq k (1+ k)) p-end))
604                (setq p (%scharcode pattern k)))
605              (when (not (eq p s))
606                (return)))))))))
607
608(defun path-str-match-p (pattern str p-start s-start p-end s-end)
609  (declare (fixnum p-start s-start p-end s-end)
610           (type simple-base-string pattern str))
611  (declare (optimize (speed 3)(safety 0)))
612  ;; does pattern match str between p-start p-end
613  (let ((esc (char-code *pathname-escape-character*)))
614    (loop     
615      (when (eq p-start p-end)
616        (return (eq s-start s-end)))
617      (when (eq s-start s-end)
618        (return nil))
619      (let ((p (%scharcode pattern p-start)))
620        (when (eq p esc)
621          (when (eq (setq p-start (1+ p-start)) p-end)
622            (return nil))
623          (setq p (%scharcode pattern p-start)))
624        (unless (eq p (%scharcode str s-start))
625          (return nil))
626        (setq p-start (1+ p-start))
627        (setq s-start (1+ s-start))))))
628     
629             
630(defun ccl-directory ()
631  (let* ((dirpath (getenv "CCL_DEFAULT_DIRECTORY")))
632    (if dirpath
633      (native-to-directory-pathname dirpath)
634      (let* ((directory-containing-heap-image
635              (make-pathname :directory (pathname-directory (%realpath (heap-image-name)))))
636             (rpath (merge-pathnames
637                     #+darwinppc-target "../Resources/ccl/"
638                     #+linux-target "Resources/ccl/"
639                     directory-containing-heap-image)))
640        (or (probe-file rpath)
641            directory-containing-heap-image)))))
642
643
644(defun user-homedir-pathname (&optional host)
645  "Return the home directory of the user as a pathname."
646  (declare (ignore host))
647  (let* ((native (or #+ccl-0711 (getenv "HOME")
648                     (get-user-home-dir (getuid)))))
649    (if native
650      (native-to-directory-pathname native))))
651
652
653
654(defloadvar *user-homedir-pathname* (user-homedir-pathname))
655
656(defun translate-logical-pathname (pathname &key)
657  "Translate PATHNAME to a physical pathname, which is returned."
658  (setq pathname (pathname pathname))
659  (let ((host (pathname-host pathname)))
660    (cond ((eq host :unspecific) pathname)
661          ((null host) (%cons-pathname (pathname-directory pathname)
662                                       (pathname-name pathname)
663                                       (pathname-type pathname)))
664          (t
665           (let ((rule (assoc pathname (logical-pathname-translations host)
666                              :test #'pathname-match-p)))  ; how can they match if hosts neq??
667             (if rule
668               (translate-logical-pathname
669                (translate-pathname pathname (car rule) (cadr rule)))
670               (signal-file-error $xnotranslation pathname)))))))
671
672;;; Hide this from COMPILE-FILE, for obscure cross-compilation reasons
673
674(defun setup-initial-translations ()
675  (setf (logical-pathname-translations "home")
676        `(("**;*.*" ,(merge-pathnames "**/*.*" (user-homedir-pathname)))))
677
678  (setf (logical-pathname-translations "ccl")
679        `(("l1;**;*.*" "ccl:level-1;**;*.*")
680          ("l1f;**;*.*" "ccl:l1-fasls;**;*.*")
681          ("ccl;*.*" ,(merge-pathnames "*.*" (ccl-directory)))
682          ("**;*.*" ,(merge-pathnames "**/*.*" (ccl-directory))))))
683
684(setup-initial-translations)
685
686
687;;; Translate the pathname; if the directory component of the result
688;;; is relative, make it absolute (relative to the current directory.)
689(defun full-pathname (path &key (no-error t))
690  (let* ((path (handler-case (translate-logical-pathname (merge-pathnames path))
691                 (error (condition) (if no-error
692                                      (return-from full-pathname nil)
693                                      (error condition)))))
694         (dir (%pathname-directory path)))
695    (if (eq (car dir) :absolute)
696      path
697      (cons-pathname (absolute-directory-list dir)
698                       (%pathname-name path)
699                       (%pathname-type path)
700                       (pathname-host path)
701                       (pathname-version path)))))
702
703
704
705
706(defparameter *module-search-path* (list
707                                    (cons-pathname '(:absolute "bin") nil nil "ccl")
708                                    (cons-pathname '(:absolute "openmcl" "modules") nil nil "home")
709                                    (cons-pathname '(:absolute "lib") nil nil "ccl")
710                                    (cons-pathname '(:absolute "library") nil nil "ccl")
711                                    (cons-pathname '(:absolute "examples" :wild-inferiors) nil nil "ccl")
712                                    (cons-pathname '(:absolute "tools") nil nil "ccl")
713                                    (cons-pathname '(:absolute "objc-bridge") nil nil "ccl")
714                                    (cons-pathname '(:absolute "cocoa-ide") nil nil "ccl"))
715  "Holds a list of pathnames to search for the file that has same name
716   as a module somebody is looking for.")
717
Note: See TracBrowser for help on using the repository browser.