All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.armedbear.lisp.StringFunctions Maven / Gradle / Ivy

There is a newer version: 1.9.2
Show newest version
/*
 * StringFunctions.java
 *
 * Copyright (C) 2003-2005 Peter Graves
 * Copyright (C) 2010 Ville Voutilainen
 * $Id$
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
 *
 * As a special exception, the copyright holders of this library give you
 * permission to link this library with independent modules to produce an
 * executable, regardless of the license terms of these independent
 * modules, and to copy and distribute the resulting executable under
 * terms of your choice, provided that you also meet, for each linked
 * independent module, the terms and conditions of the license of that
 * module.  An independent module is a module which is not derived from
 * or based on this library.  If you modify this library, you may extend
 * this exception to your version of the library, but you are not
 * obligated to do so.  If you do not wish to do so, delete this
 * exception statement from your version.
 */

package org.armedbear.lisp;

import static org.armedbear.lisp.Lisp.*;
import java.util.Arrays;
public final class StringFunctions {
    final static class StringIndicesAndChars {
        public AbstractString string1;
        public boolean convertCase = false;
        public char[] array1;
        public char[] array2;
        public int start1 = 0;
        public int end1 = 0;
        public int start2 = 0;
        public int end2 = 0;
    };
    private final static void 
        checkParams(StringIndicesAndChars indicesAndChars) {
        if (indicesAndChars.start1 < 0 
            || indicesAndChars.start1 > indicesAndChars.array1.length)
            error(new TypeError("Invalid start position " 
                                + indicesAndChars.start1 + "."));
        if (indicesAndChars.end1 < 0 
            || indicesAndChars.end1 > indicesAndChars.array1.length)
            error(new TypeError("Invalid end position " 
                                + indicesAndChars.end1 + "."));
        
        if (indicesAndChars.start1 > indicesAndChars.end1) 
            error(new TypeError("Start (" 
                                + indicesAndChars.start1 
                                + ") is greater than end (" 
                                + indicesAndChars.end1 + ")."));
        if (indicesAndChars.array2 != null) {
            if (indicesAndChars.start2 < 0 
                || indicesAndChars.start2 > indicesAndChars.array2.length)
                error(new TypeError("Invalid start2 position " 
                                    + indicesAndChars.start2 + "."));
            if (indicesAndChars.end2 < 0 
                || indicesAndChars.end2 > indicesAndChars.array2.length)
                error(new TypeError("Invalid end2 position " 
                                    + indicesAndChars.end2 + "."));
            if (indicesAndChars.start2 > indicesAndChars.end2)
                error(new TypeError("Start2 (" 
                                    + indicesAndChars.start2 
                                    + ") is greater than end2 (" 
                                    + indicesAndChars.end2 + ")."));
        }

    }
    
    private final static char upcaseIfNeeded(char c, boolean convert) {
        return convert ? LispCharacter.toUpperCase(c) : c;
    }

    final static StringIndicesAndChars
        stringIndicesAndChars(LispObject... params) {
        StringIndicesAndChars retVal = new StringIndicesAndChars();
        retVal.string1 = checkString(params[0].STRING());
        retVal.array1 = retVal.string1.getStringChars();
        retVal.end1 = retVal.array1.length;
        if (params.length == 3) {
            if (params[1] != NIL) {
                retVal.start1 = Fixnum.getValue(params[1]);
            }
            if (params[2] != NIL) {
                retVal.end1 = Fixnum.getValue(params[2]);
            }
        } else {
            retVal.array2 = params[1].STRING().getStringChars();
            retVal.end2 = retVal.array2.length;
            if (params.length > 2) {
                if (params[2] != NIL) {
                    retVal.start1 = Fixnum.getValue(params[2]);
                }
                if (params[3] != NIL) {
                    retVal.end1 = Fixnum.getValue(params[3]);
                }
                if (params[4] != NIL) {
                    retVal.start2 = Fixnum.getValue(params[4]);
                }
                if (params[5] != NIL) {
                    retVal.end2 = Fixnum.getValue(params[5]);
                }
            }
        }
        checkParams(retVal);
        return retVal;
    }

    // ### %%string=
    // Case sensitive.
    private static final Primitive __STRING_EQUAL = new pf___string_equal();
    private static final class pf___string_equal extends Primitive {
        pf___string_equal() {
            super("%%string=", PACKAGE_SYS, false);
        }

        @Override
        public LispObject execute(LispObject string1, LispObject string2)

        {
            StringIndicesAndChars chars = 
                stringIndicesAndChars(string1, string2);
            return Arrays.equals(chars.array1, chars.array2) ?
                T : NIL;
        };
    }

    // ### %string=
    // Case sensitive.
    private static final Primitive _STRING_EQUAL = new pf__string_equal();
    private static final class pf__string_equal extends Primitive {
        pf__string_equal() {
            super("%string=", PACKAGE_SYS, false);
        }

        @Override
        public LispObject execute(LispObject string1, LispObject string2,
                                  LispObject start1, LispObject end1,
                                  LispObject start2, LispObject end2)

        {
            return 
                (_STRING_NOT_EQUAL.execute(string1, string2, 
                                           start1, end1, 
                                           start2, end2)
                 == NIL) ? T : NIL;
        }
    };


    static final int notEqual(StringIndicesAndChars indicesAndChars) {
        int i = indicesAndChars.start1;
        int j = indicesAndChars.start2;
        while (true) {
            if (i == indicesAndChars.end1) {
                // Reached end of string1.
                if (j == indicesAndChars.end2)
                    return -1; // Strings are identical.
                return i;
            }
            if (j == indicesAndChars.end2) {
                // Reached end of string2 before end of string1.
                return i;
            }
            if (upcaseIfNeeded(indicesAndChars.array1[i],
                               indicesAndChars.convertCase)
                != upcaseIfNeeded(indicesAndChars.array2[j],
                                  indicesAndChars.convertCase))
                return i;
            ++i;
            ++j;
        }
    }
    // ### %string/=
    // Case sensitive.
    static final Primitive _STRING_NOT_EQUAL = new pf__string_not_equal();
    private static final class pf__string_not_equal extends Primitive {
        pf__string_not_equal() {
            super("%string/=", PACKAGE_SYS, true);
        }

        @Override
        public LispObject execute(LispObject string1, LispObject string2,
                                  LispObject start1, LispObject end1,
                                  LispObject start2, LispObject end2) {
            StringIndicesAndChars indicesAndChars = 
                stringIndicesAndChars(string1, string2, start1, end1,
                                      start2, end2);
            int tmp = notEqual(indicesAndChars);
            return (tmp >= 0) ? Fixnum.getInstance(tmp) : NIL;
        }
    };

    // ### %string-equal
    // Case insensitive.
    private static final Primitive _STRING_EQUAL_IGNORE_CASE = new pf__string_equal_ignore_case();
    private static final class pf__string_equal_ignore_case extends Primitive {
        pf__string_equal_ignore_case() {
            super("%string-equal", PACKAGE_SYS, true);
        }

        @Override
        public LispObject execute(LispObject string1, LispObject string2,
                                  LispObject start1, LispObject end1,
                                  LispObject start2, LispObject end2)

        {
            return (_STRING_NOT_EQUAL_IGNORE_CASE.execute(string1, string2, 
                                                          start1, end1, 
                                                          start2, end2) 
                    == NIL) ? T : NIL;
        }
    };

    // ### %string-not-equal
    // Case insensitive.
    static final Primitive _STRING_NOT_EQUAL_IGNORE_CASE = new pf__string_not_equal_ignore_case();
    private static final class pf__string_not_equal_ignore_case extends Primitive {
        pf__string_not_equal_ignore_case() {
            super("%string-not-equal", PACKAGE_SYS, true);
        }

        @Override
        public LispObject execute(LispObject string1, LispObject string2,
                                  LispObject start1, LispObject end1,
                                  LispObject start2, LispObject end2) {
            StringIndicesAndChars indicesAndChars = 
                stringIndicesAndChars(string1, string2, start1, end1,
                                      start2, end2);
            indicesAndChars.convertCase = true;
            int tmp = notEqual(indicesAndChars);
            return (tmp >= 0) ? Fixnum.getInstance(tmp) : NIL;
        }
    };

    static final int lessThan(StringIndicesAndChars indicesAndChars) {
        int i = indicesAndChars.start1;
        int j = indicesAndChars.start2;
        while (true) {
            if (i == indicesAndChars.end1) {
                // Reached end of string1.
                if (j == indicesAndChars.end2)
                    return -1; // Strings are identical.
                return i;
            }
            if (j == indicesAndChars.end2) {
                // Reached end of string2.
                return -1;
            }
            char c1 = upcaseIfNeeded(indicesAndChars.array1[i], 
                                     indicesAndChars.convertCase);
            char c2 = upcaseIfNeeded(indicesAndChars.array2[j],
                                     indicesAndChars.convertCase);
            if (c1 == c2) {
                ++i;
                ++j;
                continue;
            }
            if (c1 < c2)
                return (i);
            // c1 > c2
            return -1;
        }
    }

    // ### %string<
    // Case sensitive.
    private static final Primitive _STRING_LESS_THAN = new pf__string_less_than();
    private static final class pf__string_less_than extends Primitive {
        pf__string_less_than() {
            super("%string<", PACKAGE_SYS, true);
        }

        @Override
        public LispObject execute(LispObject string1, LispObject string2,
                                  LispObject start1, LispObject end1,
                                  LispObject start2, LispObject end2) {
            StringIndicesAndChars indicesAndChars = 
                stringIndicesAndChars(string1, string2, 
                                      start1, end1, start2, end2);
            int retVal = lessThan(indicesAndChars);
            return (retVal >= 0) ? Fixnum.getInstance(retVal) : NIL;
        }
    };

    static LispObject 
        swapReturnValue(int original,
                        StringIndicesAndChars indicesAndChars) {
        if (original < 0) {
            return NIL;
        }
        int delta = original - indicesAndChars.start1;
        int retVal = indicesAndChars.start2 + delta;
        return Fixnum.getInstance(retVal);
    }

    // ### %string>
    // Case sensitive.
    private static final Primitive _STRING_GREATER_THAN = new pf__string_greater_than();
    private static final class pf__string_greater_than extends Primitive {
        pf__string_greater_than() {
            super("%string>", PACKAGE_SYS, true);
        }

        @Override
        public LispObject execute(LispObject string1, LispObject string2,
                                  LispObject start1, LispObject end1,
                                  LispObject start2, LispObject end2) {
            // note the swap of the strings and lengths here..
            StringIndicesAndChars indicesAndChars = 
                stringIndicesAndChars(string2, string1, 
                                      start2, end2,
                                      start1, end1);
            int tmp = lessThan(indicesAndChars);
            return swapReturnValue(tmp, indicesAndChars);
        }
    };

    static final int lessThanOrEqual(StringIndicesAndChars indicesAndChars) {
        int i = indicesAndChars.start1;
        int j = indicesAndChars.start2;
        while (true) {
            if (i == indicesAndChars.end1) {
                // Reached end of string1.
                return i;
            }
            if (j == indicesAndChars.end2) {
                // Reached end of string2.
                return -1;
            }
            char c1 = upcaseIfNeeded(indicesAndChars.array1[i], 
                                     indicesAndChars.convertCase);
            char c2 = upcaseIfNeeded(indicesAndChars.array2[j],
                                     indicesAndChars.convertCase);
            if (c1 == c2) {
                ++i;
                ++j;
                continue;
            }
            if (c1 > c2)
                return -1;
            // c1 < c2
            return (i);
        }
    }
    // ### %string<=
    // Case sensitive.
    private static final Primitive _STRING_LE = new pf__string_le();
    private static final class pf__string_le extends Primitive {
        pf__string_le() {
            super("%string<=", PACKAGE_SYS, true);
        }

        @Override
        public LispObject execute(LispObject string1, LispObject string2,
                                  LispObject start1, LispObject end1,
                                  LispObject start2, LispObject end2) {

            StringIndicesAndChars indicesAndChars = 
                stringIndicesAndChars(string1, string2, 
                                      start1, end1, start2, end2);
            int retVal = lessThanOrEqual(indicesAndChars);
            return (retVal >= 0) ? Fixnum.getInstance(retVal) : NIL;
        }
    };

    // ### %string>=
    // Case sensitive.
    private static final Primitive _STRING_GE = new pf__string_ge();
    private static final class pf__string_ge extends Primitive {
        pf__string_ge() {
            super("%string>=", PACKAGE_SYS, true);
        }

        @Override
        public LispObject execute(LispObject string1, LispObject string2,
                                  LispObject start1, LispObject end1,
                                  LispObject start2, LispObject end2) {
            // note the swap of the strings and lengths here..
            StringIndicesAndChars indicesAndChars = 
                stringIndicesAndChars(string2, string1,
                                      start2, end2,
                                      start1, end1);
            int tmp = lessThanOrEqual(indicesAndChars);
            return swapReturnValue(tmp, indicesAndChars);
        }
    };


    // ### %string-lessp
    // Case insensitive.
    private static final Primitive _STRING_LESSP = new pf__string_lessp();
    private static final class pf__string_lessp extends Primitive {
        pf__string_lessp() {
            super("%string-lessp", PACKAGE_SYS, true);
        }

        @Override
        public LispObject execute(LispObject string1, LispObject string2,
                                  LispObject start1, LispObject end1,
                                  LispObject start2, LispObject end2) {
            StringIndicesAndChars indicesAndChars = 
                stringIndicesAndChars(string1, string2, 
                                      start1, end1, start2, end2);
            indicesAndChars.convertCase = true;
            int retVal = lessThan(indicesAndChars);
            return (retVal >= 0) ? Fixnum.getInstance(retVal) : NIL;
        }
    };

    // ### %string-greaterp
    // Case insensitive.
    private static final Primitive _STRING_GREATERP = new pf__string_greaterp();
    private static final class pf__string_greaterp extends Primitive {
        pf__string_greaterp() {
            super("%string-greaterp", PACKAGE_SYS, true);
        }

        @Override
        public LispObject execute(LispObject string1, LispObject string2,
                                  LispObject start1, LispObject end1,
                                  LispObject start2, LispObject end2) {
            // note the swap of the strings and lengths here..
            StringIndicesAndChars indicesAndChars = 
                stringIndicesAndChars(string2, string1,
                                      start2, end2,
                                      start1, end1);
            indicesAndChars.convertCase = true;
            int tmp = lessThan(indicesAndChars);
            return swapReturnValue(tmp, indicesAndChars);
        }
    };
    // ### %string-not-lessp
    // Case insensitive.
    private static final Primitive _STRING_NOT_LESSP = new pf__string_not_lessp();
    private static final class pf__string_not_lessp extends Primitive {
        pf__string_not_lessp() {
            super("%string-not-lessp", PACKAGE_SYS, true);
        }

        @Override
        public LispObject execute(LispObject string1, LispObject string2,
                                  LispObject start1, LispObject end1,
                                  LispObject start2, LispObject end2) {
            // note the swap of the strings and lengths here..
            StringIndicesAndChars indicesAndChars = 
                stringIndicesAndChars(string2, string1,
                                      start2, end2,
                                      start1, end1);
            indicesAndChars.convertCase = true;
            int tmp = lessThanOrEqual(indicesAndChars);
            return swapReturnValue(tmp, indicesAndChars);
        }
    };

    // ### %string-not-greaterp
    // Case insensitive.
    private static final Primitive _STRING_NOT_GREATERP = new pf__string_not_greaterp();
    private static final class pf__string_not_greaterp extends Primitive {
        pf__string_not_greaterp() {
            super("%string-not-greaterp", PACKAGE_SYS, true);
        }

        @Override
        public LispObject execute(LispObject string1, LispObject string2,
                                  LispObject start1, LispObject end1,
                                  LispObject start2, LispObject end2) {
            StringIndicesAndChars indicesAndChars = 
                stringIndicesAndChars(string1, string2,
                                      start1, end1,
                                      start2, end2);
            indicesAndChars.convertCase = true;
            int tmp = lessThanOrEqual(indicesAndChars);
            return (tmp >= 0) ? Fixnum.getInstance(tmp) : NIL;
        }
    };

    // ### %string-upcase
    private static final Primitive _STRING_UPCASE = new pf__string_upcase();
    private static final class pf__string_upcase extends Primitive {
        pf__string_upcase() {
            super("%string-upcase", PACKAGE_SYS, true);
        }

        @Override
        public LispObject execute(LispObject string, LispObject start,
                                  LispObject end)

        {
            StringIndicesAndChars indicesAndChars = 
                stringIndicesAndChars(string, start, end);
            char[] array = new char[indicesAndChars.array1.length];
            System.arraycopy(indicesAndChars.array1, 0,
                             array, 0,
                             indicesAndChars.start1);
            for (int i = indicesAndChars.start1; i < indicesAndChars.end1; i++)
                array[i] = LispCharacter.toUpperCase(indicesAndChars.array1[i]);
            System.arraycopy(indicesAndChars.array1, indicesAndChars.end1,
                      array, indicesAndChars.end1,
                             indicesAndChars.array1.length 
                             - indicesAndChars.end1);
            return new SimpleString(array);
        }
    };

    // ### %string-downcase
    private static final Primitive _STRING_DOWNCASE = new pf__string_downcase();
    private static final class pf__string_downcase extends Primitive {
        pf__string_downcase() {
            super("%string-downcase", PACKAGE_SYS, true);
        }

        @Override
        public LispObject execute(LispObject string, LispObject start,
                                  LispObject end) {
            StringIndicesAndChars indicesAndChars = 
                stringIndicesAndChars(string, start, end);
            char[] array = new char[indicesAndChars.array1.length];
            System.arraycopy(indicesAndChars.array1, 0,
                             array, 0,
                             indicesAndChars.start1);
            for (int i = indicesAndChars.start1; i < indicesAndChars.end1; i++)
                array[i] = LispCharacter.toLowerCase(indicesAndChars.array1[i]);
            System.arraycopy(indicesAndChars.array1, indicesAndChars.end1,
                      array, indicesAndChars.end1,
                             indicesAndChars.array1.length 
                             - indicesAndChars.end1);
            return new SimpleString(array);
        }
    };

    // ### %string-capitalize
    private static final Primitive _STRING_CAPITALIZE = new pf__string_capitalize();
    private static final class pf__string_capitalize extends Primitive {
        pf__string_capitalize() {
            super("%string-capitalize", PACKAGE_SYS, true);
        }

        @Override
        public LispObject execute(LispObject string, LispObject start,
                                  LispObject end)

        {
            StringIndicesAndChars indicesAndChars = 
                stringIndicesAndChars(string, start, end);
            char[] array = new char[indicesAndChars.array1.length];
            boolean lastCharWasAlphanumeric = false;
            System.arraycopy(indicesAndChars.array1, 0,
                             array, 0,
                             indicesAndChars.start1);
            for (int i = indicesAndChars.start1; 
                 i < indicesAndChars.end1; i++) {
                char c = indicesAndChars.array1[i];
                if (Character.isLowerCase(c)) {
                    array[i] = lastCharWasAlphanumeric ? 
                        c : LispCharacter.toUpperCase(c);
                    lastCharWasAlphanumeric = true;
                } else if (Character.isUpperCase(c)) {
                    array[i] = lastCharWasAlphanumeric ? 
                        LispCharacter.toLowerCase(c) : c;
                    lastCharWasAlphanumeric = true;
                } else {
                    array[i] = c;
                    lastCharWasAlphanumeric = Character.isDigit(c);
                }
            }
            System.arraycopy(indicesAndChars.array1, indicesAndChars.end1,
                      array, indicesAndChars.end1,
                             indicesAndChars.array1.length 
                             - indicesAndChars.end1);
            return new SimpleString(array);
        }
    };

    // ### %nstring-upcase
    private static final Primitive _NSTRING_UPCASE = new pf__nstring_upcase();
    private static final class pf__nstring_upcase extends Primitive {
        pf__nstring_upcase() {
            super("%nstring-upcase", PACKAGE_SYS, true);
        }

        @Override
        public LispObject execute(LispObject string, LispObject start,
                                  LispObject end)

        {
            StringIndicesAndChars indicesAndChars = 
                stringIndicesAndChars(string, start, end);
            AbstractString retString = indicesAndChars.string1; 
            for (int i = indicesAndChars.start1; i < indicesAndChars.end1; i++) 
                retString.setCharAt(i,
                                    LispCharacter.
                                    toUpperCase(
                                                retString.charAt(i)));
            return retString;
        }
    };

    // ### %nstring-downcase
    private static final Primitive _NSTRING_DOWNCASE = new pf__nstring_downcase();
    private static final class pf__nstring_downcase extends Primitive {
        pf__nstring_downcase() {
            super("%nstring-downcase", PACKAGE_SYS, true);
        }

        @Override
        public LispObject execute(LispObject string, LispObject start,
                                  LispObject end)

        {
            StringIndicesAndChars indicesAndChars = 
                stringIndicesAndChars(string, start, end);
            AbstractString retString = indicesAndChars.string1; 
            for (int i = indicesAndChars.start1; i < indicesAndChars.end1; i++)
                retString.setCharAt(i,
                                    LispCharacter.
                                    toLowerCase(retString.charAt(i)));
            return retString;
        }
    };

    // ### %nstring-capitalize
    private static final Primitive _NSTRING_CAPITALIZE = new pf__nstring_capitalize();
    private static final class pf__nstring_capitalize extends Primitive {
        pf__nstring_capitalize() {
            super("%nstring-capitalize", PACKAGE_SYS, true);
        }

        @Override
        public LispObject execute(LispObject string, LispObject start,
                                  LispObject end)

        {
            StringIndicesAndChars indicesAndChars = 
                stringIndicesAndChars(string, start, end);
            boolean lastCharWasAlphanumeric = false;
            AbstractString retString = indicesAndChars.string1; 
            for (int i = indicesAndChars.start1; 
                 i < indicesAndChars.end1; i++) {
                char c = retString.charAt(i);
                if (Character.isLowerCase(c)) {
                    if (!lastCharWasAlphanumeric)
                        retString.setCharAt(i,
                                            LispCharacter.toUpperCase(c));
                    lastCharWasAlphanumeric = true;
                } else if (Character.isUpperCase(c)) {
                    if (lastCharWasAlphanumeric)
                        retString.setCharAt(i,
                                            LispCharacter.toLowerCase(c));
                    lastCharWasAlphanumeric = true;
                } else
                    lastCharWasAlphanumeric = Character.isDigit(c);
            }
            return retString;
        }
    };

    // ### stringp
    public static final Primitive STRINGP = new pf_stringp();
    private static final class pf_stringp extends Primitive {
        pf_stringp() {
            super("stringp", "object");
        }

        @Override
        public LispObject execute(LispObject arg) {
            return arg.STRINGP();
        }
    };

    // ### simple-string-p
    public static final Primitive SIMPLE_STRING_P = new pf_simple_string_p();
    private static final class pf_simple_string_p extends Primitive {
        pf_simple_string_p() {
            super("simple-string-p", "object");
        }

        @Override
        public LispObject execute(LispObject arg) {
            return arg.SIMPLE_STRING_P();
        }
    };

    // ### %make-string
    // %make-string size initial-element element-type => string
    // Returns a simple string.
    private static final Primitive _MAKE_STRING = new pf__make_string();
    private static final class pf__make_string extends Primitive {
        pf__make_string() {
            super("%make-string", PACKAGE_SYS, false);
        }

        @Override
        public LispObject execute(LispObject size, LispObject initialElement,
                                  LispObject elementType)

        {
            final int n = Fixnum.getValue(size);
            if (n < 0 || n >= ARRAY_DIMENSION_MAX) {
                StringBuilder sb = new StringBuilder();
                sb.append("The size specified for this string (");
                sb.append(n);
                sb.append(')');
                if (n >= ARRAY_DIMENSION_MAX) {
                    sb.append(" is >= ARRAY-DIMENSION-LIMIT (");
                    sb.append(ARRAY_DIMENSION_MAX);
                    sb.append(").");
                } else
                    sb.append(" is negative.");
                return error(new LispError(sb.toString()));
            }
            // Ignore elementType.
            SimpleString string = new SimpleString(n);
            if (initialElement != NIL) {
                // Initial element was specified.
                char c = checkCharacter(initialElement).getValue();
                string.fill(c);
            }
            return string;
        }
    };

    // ### char
    private static final Primitive CHAR = new pf_char();
    private static final class pf_char extends Primitive {
        pf_char() {
            super(Symbol.CHAR, "string index");
        }

        @Override
        public LispObject execute(LispObject first, LispObject second)

        {
            return first.CHAR(Fixnum.getValue(second));
        }
    };

    // ### schar
    private static final Primitive SCHAR = new pf_schar();
    private static final class pf_schar extends Primitive {
        pf_schar() {
            super(Symbol.SCHAR, "string index");
        }

        @Override
        public LispObject execute(LispObject first, LispObject second)

        {
            return first.SCHAR(Fixnum.getValue(second));
        }
    };

    // ### set-char
    private static final Primitive SET_CHAR = new pf_set_char();
    private static final class pf_set_char extends Primitive {
        pf_set_char() {
            super(Symbol.SET_CHAR, "string index character");
        }

        @Override
        public LispObject execute(LispObject first, LispObject second,
                                  LispObject third)

        {
            checkString(first).setCharAt(Fixnum.getValue(second),
                                         LispCharacter.getValue(third));
            return third;
        }
    };

    // ### set-schar
    private static final Primitive SET_SCHAR = new pf_set_schar();
    private static final class pf_set_schar extends Primitive {
        pf_set_schar() {
            super(Symbol.SET_SCHAR, "string index character");
        }

        @Override
        public LispObject execute(LispObject first, LispObject second,
                                  LispObject third)

        {
            if (first instanceof SimpleString) {
                ((SimpleString)first).setCharAt(Fixnum.getValue(second),
                                                LispCharacter.getValue(third));
                return third;
            }
            return type_error(first, Symbol.SIMPLE_STRING);
        }
    };

    // ### string-position
    private static final Primitive STRING_POSITION = new pf_string_position();
    private static final class pf_string_position extends Primitive {
        pf_string_position() {
            super("string-position", PACKAGE_EXT, true);
        }

        @Override
        public LispObject execute(LispObject first, LispObject second,
                                  LispObject third)

        {
            char c = LispCharacter.getValue(first);
            AbstractString string = checkString(second);
            int start = Fixnum.getValue(third);
            for (int i = start, limit = string.length(); i < limit; i++) {
                if (string.charAt(i) == c)
                    return number(i);
            }
            return NIL;
        }
    };

    // ### string-find
    private static final Primitive STRING_FIND = new pf_string_find();
    private static final class pf_string_find extends Primitive {
        pf_string_find() {
            super("string-find", PACKAGE_EXT, true, "char string");
        }

        @Override
        public LispObject execute(LispObject first, LispObject second)

        {
            if (first instanceof LispCharacter) {
                final char c = ((LispCharacter)first).value;
                AbstractString string = Lisp.checkString(second);
                final int limit = string.length();
                for (int i = 0; i < limit; i++) {
                    if (string.charAt(i) == c)
                        return first;
                }
            }
            return NIL;
        }
    };

    // ### simple-string-search pattern string => position
    // Searches string for a substring that matches pattern.
    private static final Primitive SIMPLE_STRING_SEARCH = new pf_simple_string_search();
    private static final class pf_simple_string_search extends Primitive {
        pf_simple_string_search() {
            super("simple-string-search", PACKAGE_EXT, true);
        }

        @Override
        public LispObject execute(LispObject first, LispObject second)

        {
            // FIXME Don't call getStringValue() here! (Just look at the chars.)
            int index = second.getStringValue().indexOf(first.getStringValue());
            return index >= 0 ? Fixnum.getInstance(index) : NIL;
        }
    };

    // ### simple-string-fill string character => string
    private static final Primitive STRING_FILL = new pf_string_fill();
    private static final class pf_string_fill extends Primitive {
        pf_string_fill() {
            super("simple-string-fill", PACKAGE_EXT, true);
        }

        @Override
        public LispObject execute(LispObject first, LispObject second)

        {
            if (first instanceof AbstractString) {
                AbstractString s = (AbstractString) first;
                s.fill(LispCharacter.getValue(second));
                return first;
            }
            return type_error(first, Symbol.SIMPLE_STRING);
        }
    };

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy