X-Git-Url: http://repo.macrolet.net/gitweb/?a=blobdiff_plain;f=src%2Fcode%2Fstring.lisp;h=72df091f15653f817c09968df8502e6cb0619faf;hb=cee8ef591040db9a79cdd19297867672a9529051;hp=36bc060d45ada5ff856f6d6495f5da5bcb385363;hpb=2de1b72f4bec82ad5289f33a84b34fe9cb62bd0a;p=sbcl.git diff --git a/src/code/string.lisp b/src/code/string.lisp index 36bc060..72df091 100644 --- a/src/code/string.lisp +++ b/src/code/string.lisp @@ -9,6 +9,9 @@ (in-package "SB!IMPL") +(eval-when (:compile-toplevel) + (sb!xc:defmacro %string (x) `(if (stringp ,x) ,x (string ,x)))) + (defun string (x) #!+sb-doc "Coerces X into a string. If X is a string, X is returned. If X is a @@ -37,26 +40,18 @@ ;;; keywords are parsed, and the string is hacked into a ;;; simple-string. (sb!xc:defmacro with-one-string ((string start end) &body forms) - `(let* ((,string (if (stringp ,string) ,string (string ,string)))) + `(let ((,string (%string ,string))) (with-array-data ((,string ,string) (,start ,start) (,end ,end) :check-fill-pointer t) ,@forms))) -;;; WITH-STRING is like WITH-ONE-STRING, but doesn't parse keywords. -(sb!xc:defmacro with-string (string &rest forms) - `(let ((,string (if (stringp ,string) ,string (string ,string)))) - (with-array-data ((,string ,string) - (start) - (end) - :check-fill-pointer t) - ,@forms))) ;;; WITH-TWO-STRINGS is used to set up string comparison operations. The ;;; keywords are parsed, and the strings are hacked into SIMPLE-STRINGs. (sb!xc:defmacro with-two-strings (string1 string2 start1 end1 cum-offset-1 start2 end2 &rest forms) - `(let ((,string1 (if (stringp ,string1) ,string1 (string ,string1))) - (,string2 (if (stringp ,string2) ,string2 (string ,string2)))) + `(let ((,string1 (%string ,string1)) + (,string2 (%string ,string2))) (with-array-data ((,string1 ,string1 :offset-var ,cum-offset-1) (,start1 ,start1) (,end1 ,end1) @@ -404,36 +399,40 @@ new string COUNT long filled with the fill character." (%capitalize string start end)) ) ; FLET +(defun generic-string-trim (char-bag string left-p right-p) + (let ((header (%string string))) + (with-array-data ((string header) + (start) + (end) + :check-fill-pointer t) + (let* ((left-end (if left-p + (do ((index start (1+ index))) + ((or (= index (the fixnum end)) + (not (find (schar string index) + char-bag + :test #'char=))) + index) + (declare (fixnum index))) + start)) + (right-end (if right-p + (do ((index (1- (the fixnum end)) (1- index))) + ((or (< index left-end) + (not (find (schar string index) + char-bag + :test #'char=))) + (1+ index)) + (declare (fixnum index))) + end))) + (if (and (eql left-end start) + (eql right-end end)) + header + (subseq (the simple-string string) left-end right-end)))))) + (defun string-left-trim (char-bag string) - (with-string string - (do ((index start (1+ index))) - ((or (= index (the fixnum end)) - (not (find (schar string index) char-bag :test #'char=))) - (subseq (the simple-string string) index end)) - (declare (fixnum index))))) + (generic-string-trim char-bag string t nil)) (defun string-right-trim (char-bag string) - (with-string string - (do ((index (1- (the fixnum end)) (1- index))) - ((or (< index start) - (not (find (schar string index) char-bag :test #'char=))) - (subseq (the simple-string string) start (1+ index))) - (declare (fixnum index))))) + (generic-string-trim char-bag string nil t)) (defun string-trim (char-bag string) - (with-string string - (let* ((left-end (do ((index start (1+ index))) - ((or (= index (the fixnum end)) - (not (find (schar string index) - char-bag - :test #'char=))) - index) - (declare (fixnum index)))) - (right-end (do ((index (1- (the fixnum end)) (1- index))) - ((or (< index left-end) - (not (find (schar string index) - char-bag - :test #'char=))) - (1+ index)) - (declare (fixnum index))))) - (subseq (the simple-string string) left-end right-end)))) + (generic-string-trim char-bag string t t))