source: trunk/ccl/level-1/l1-pathnames.lisp @ 6941

Last change on this file since 6941 was 6941, checked in by gb, 14 years ago

Add objc-bridge, cocoa-ide to *MODULE-SEARCH-PATH*.

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