source: release/1.6/source/level-1/l1-pathnames.lisp @ 15888

Last change on this file since 15888 was 14344, checked in by rme, 9 years ago

Don't call truename in user-homedir-pathname.

If we do, we choke when *default-pathname-defaults* is non-empty.

See http://groups.google.com/group/comp.lang.lisp/msg/4339e788b4039faa

(The use of truename was introduced back in r10426.)

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