Imported from ../bash-2.05b.tar.gz.
This commit is contained in:
		
					parent
					
						
							
								f73dda092b
							
						
					
				
			
			
				commit
				
					
						7117c2d221
					
				
			
		
					 362 changed files with 34387 additions and 15063 deletions
				
			
		|  | @ -39,6 +39,8 @@ MV = mv | |||
| 
 | ||||
| SHELL = @MAKE_SHELL@ | ||||
| 
 | ||||
| PROFILE_FLAGS = @PROFILE_FLAGS@ | ||||
| 
 | ||||
| CFLAGS = @CFLAGS@ | ||||
| LOCAL_CFLAGS = @LOCAL_CFLAGS@ | ||||
| CPPFLAGS = @CPPFLAGS@ | ||||
|  | @ -51,7 +53,8 @@ BASHINCDIR = ${topdir}/include | |||
| 
 | ||||
| INCLUDES = -I. -I../.. -I$(topdir) -I$(BASHINCDIR) -I$(topdir)/lib | ||||
| 
 | ||||
| CCFLAGS = $(DEFS) $(LOCAL_DEFS) $(CPPFLAGS) ${INCLUDES} $(LOCAL_CFLAGS) $(CFLAGS) | ||||
| CCFLAGS = $(PROFILE_FLAGS) $(DEFS) $(LOCAL_DEFS) $(CPPFLAGS) ${INCLUDES} \
 | ||||
| 	  $(LOCAL_CFLAGS) $(CFLAGS) | ||||
| 
 | ||||
| # Here is a rule for making .o files from .c files that doesn't force
 | ||||
| # the type of the machine (like -sun3) into the flags.
 | ||||
|  | @ -62,12 +65,13 @@ CCFLAGS = $(DEFS) $(LOCAL_DEFS) $(CPPFLAGS) ${INCLUDES} $(LOCAL_CFLAGS) $(CFLAGS | |||
| LIBRARY_NAME = libglob.a | ||||
| 
 | ||||
| # The C code source files for this library.
 | ||||
| CSOURCES = $(srcdir)/glob.c $(srcdir)/strmatch.c | ||||
| CSOURCES = $(srcdir)/glob.c $(srcdir)/strmatch.c $(srcdir)/smatch.c \
 | ||||
| 	   $(srcdir)/xmbsrtowcs.c | ||||
| 
 | ||||
| # The header files for this library.
 | ||||
| HSOURCES = $(srcdir)/strmatch.h | ||||
| 
 | ||||
| OBJECTS = glob.o strmatch.o | ||||
| OBJECTS = glob.o strmatch.o smatch.o xmbsrtowcs.o | ||||
| 
 | ||||
| # The texinfo files which document this library.
 | ||||
| DOCSOURCE = doc/glob.texi | ||||
|  | @ -121,15 +125,34 @@ mostlyclean: clean | |||
| #								     #
 | ||||
| ######################################################################
 | ||||
| 
 | ||||
| smatch.o: strmatch.h | ||||
| smatch.o: $(BUILD_DIR)/config.h | ||||
| smatch.o: $(BASHINCDIR)/chartypes.h | ||||
| smatch.o: $(BASHINCDIR)/ansi_stdlib.h $(topdir)/bashansi.h | ||||
| smatch.o: $(BASHINCDIR)/shmbutil.h | ||||
| smatch.o: $(topdir)/xmalloc.h | ||||
| 
 | ||||
| strmatch.o: strmatch.h | ||||
| strmatch.o: $(BUILD_DIR)/config.h | ||||
| strmatch.o: $(BASHINCDIR)/chartypes.h | ||||
| strmatch.o: $(BASHINCDIR)/stdc.h | ||||
| 
 | ||||
| glob.o: $(BUILD_DIR)/config.h | ||||
| glob.o: $(topdir)/bashtypes.h $(BASHINCDIR)/ansi_stdlib.h $(topdir)/bashansi.h | ||||
| glob.o: $(BASHINCDIR)/posixstat.h $(BASHINCDIR)/memalloc.h | ||||
| glob.o: strmatch.h | ||||
| glob.o: strmatch.h glob.h | ||||
| glob.o: $(BASHINCDIR)/shmbutil.h | ||||
| glob.o: $(topdir)/xmalloc.h | ||||
| 
 | ||||
| xmbsrtowcs.o: ${BUILD_DIR}/config.h | ||||
| xmbsrtowcs.o: ${topdir}/bashansi.h ${BASHINCDIR}/ansi_stdlib.h | ||||
| xmbsrtowcs.o: ${BASHINCDIR}/shmbutil.h | ||||
| 
 | ||||
| # Rules for deficient makes, like SunOS and Solaris
 | ||||
| strmatch.o: strmatch.c | ||||
| glob.o: glob.c | ||||
| strmatch.o: strmatch.c | ||||
| smatch.o: smatch.c | ||||
| xmbsrtowcs.o: xmbsrtowcs.c | ||||
| 
 | ||||
| # dependencies for C files that include other C files
 | ||||
| glob.o: glob_loop.c | ||||
| smatch.o: sm_loop.c | ||||
|  |  | |||
|  | @ -1,141 +1,140 @@ | |||
| /* collsyms.h -- collating symbol names and their corresponding characters
 | ||||
| 		 (in ascii) as given by POSIX.2 in table 2.8. */ | ||||
| 
 | ||||
| /* Copyright (C) 1997 Free Software Foundation, Inc.
 | ||||
| /* Copyright (C) 1997-2002 Free Software Foundation, Inc.
 | ||||
| 
 | ||||
|    This file is part of GNU Bash, the Bourne Again SHell. | ||||
|     | ||||
| 
 | ||||
|    Bash 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, or (at your option) any later | ||||
|    version. | ||||
|                | ||||
| 
 | ||||
|    Bash 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 Bash; see the file COPYING.  If not, write to the Free Software | ||||
|    Foundation, 59 Temple Place, Suite 330, Boston, MA 02111 USA. */ | ||||
| 
 | ||||
| #ifndef _COLLSYMS_H_ | ||||
| #  define _COLLSYSMS_H_ | ||||
| 
 | ||||
| /* The upper-case letters, lower-case letters, and digits are omitted from
 | ||||
|    this table.  The digits are not included in the table in the POSIX.2 | ||||
|    spec.  The upper and lower case letters are translated by the code | ||||
|    in fnmatch.c:collsym(). */ | ||||
|    in smatch.c:collsym(). */ | ||||
| 
 | ||||
| typedef struct _collsym { | ||||
|   char *name; | ||||
|   char code; | ||||
| } COLLSYM; | ||||
| typedef struct _COLLSYM { | ||||
|   XCHAR *name; | ||||
|   CHAR code; | ||||
| } __COLLSYM; | ||||
| 
 | ||||
| static COLLSYM posix_collsyms[] = | ||||
| static __COLLSYM POSIXCOLL [] = | ||||
| { | ||||
| 	{  "NUL",		'\0' }, | ||||
| 	{  "SOH",		'\001' }, | ||||
| 	{  "STX",		'\002' }, | ||||
| 	{  "ETX",		'\003' }, | ||||
| 	{  "EOT",		'\004' }, | ||||
| 	{  "ENQ",		'\005' }, | ||||
| 	{  "ACK",		'\006' }, | ||||
| 	{  L("NUL"),			L('\0') }, | ||||
| 	{  L("SOH"),			L('\001') }, | ||||
| 	{  L("STX"),			L('\002') }, | ||||
| 	{  L("ETX"),			L('\003') }, | ||||
| 	{  L("EOT"),			L('\004') }, | ||||
| 	{  L("ENQ"),			L('\005') }, | ||||
| 	{  L("ACK"),			L('\006') }, | ||||
| #ifdef __STDC__ | ||||
| 	{  "alert",		'\a' }, | ||||
| 	{  L("alert"),			L('\a') }, | ||||
| #else | ||||
| 	{  "alert",		'\007' }, | ||||
| 	{  L("alert"),			L('\007') }, | ||||
| #endif | ||||
| 	{  "BS",		'\010' }, | ||||
| 	{  "backspace",		'\b' }, | ||||
| 	{  "HT",		'\011' }, | ||||
| 	{  "tab",		'\t' }, | ||||
| 	{  "LF",		'\012' }, | ||||
| 	{  "newline",		'\n' }, | ||||
| 	{  "VT",		'\013' }, | ||||
| 	{  "vertical-tab",	'\v' }, | ||||
| 	{  "FF",		'\014' }, | ||||
| 	{  "form-feed",		'\f' }, | ||||
| 	{  "CR",		'\015' }, | ||||
| 	{  "carriage-return",	'\r' }, | ||||
| 	{  "SO",		'\016' }, | ||||
| 	{  "SI",		'\017' }, | ||||
| 	{  "DLE",		'\020' }, | ||||
| 	{  "DC1",		'\021' }, | ||||
| 	{  "DC2",		'\022' }, | ||||
| 	{  "DC3",		'\023' }, | ||||
| 	{  "DC4",		'\024' }, | ||||
| 	{  "NAK",		'\025' }, | ||||
| 	{  "SYN",		'\026' }, | ||||
| 	{  "ETB",		'\027' }, | ||||
| 	{  "CAN",		'\030' }, | ||||
| 	{  "EM",		'\031' }, | ||||
| 	{  "SUB",		'\032' }, | ||||
| 	{  "ESC",		'\033' }, | ||||
| 	{  "IS4",		'\034' }, | ||||
| 	{  "FS",		'\034' }, | ||||
| 	{  "IS3",		'\035' }, | ||||
| 	{  "GS",		'\035' }, | ||||
| 	{  "IS2",		'\036' }, | ||||
| 	{  "RS",		'\036' }, | ||||
| 	{  "IS1",		'\037' }, | ||||
| 	{  "US",		'\037' }, | ||||
| 	{  "space",		' ' }, | ||||
| 	{  "exclamation-mark",	'!' }, | ||||
| 	{  "quotation-mark",	'"' }, | ||||
| 	{  "number-sign",	'#' }, | ||||
| 	{  "dollar-sign",	'$' }, | ||||
| 	{  "percent-sign",	'%' }, | ||||
| 	{  "ampersand",		'&' }, | ||||
| 	{  "apostrophe",	'\'' }, | ||||
| 	{  "left-parenthesis",	'(' }, | ||||
| 	{  "right-parenthesis",	')' }, | ||||
| 	{  "asterisk",		'*' }, | ||||
| 	{  "plus-sign",		'+' }, | ||||
| 	{  "comma",		',' }, | ||||
| 	{  "hyphen",		'-' }, | ||||
| 	{  "hyphen-minus",	'-' }, | ||||
| 	{  "minus",		'-' },		/* extension from POSIX.2 */ | ||||
| 	{  "dash",		'-' },		/* extension from POSIX.2 */ | ||||
| 	{  "period",		'.' }, | ||||
| 	{  "full-stop",		'.' }, | ||||
| 	{  "slash",		'/' }, | ||||
| 	{  "solidus",		'/' },		/* extension from POSIX.2 */ | ||||
| 	{  "zero",		'0' }, | ||||
| 	{  "one",		'1' }, | ||||
| 	{  "two",		'2' }, | ||||
| 	{  "three",		'3' }, | ||||
| 	{  "four",		'4' }, | ||||
| 	{  "five",		'5' }, | ||||
| 	{  "six",		'6' }, | ||||
| 	{  "seven",		'7' }, | ||||
| 	{  "eight",		'8' }, | ||||
| 	{  "nine",		'9' }, | ||||
| 	{  "colon",		':' }, | ||||
| 	{  "semicolon",		';' }, | ||||
| 	{  "less-than-sign",	'<' }, | ||||
| 	{  "equals-sign",	'=' }, | ||||
| 	{  "greater-than-sign",	'>' }, | ||||
| 	{  "question-mark",	'?' }, | ||||
| 	{  "commercial-at",	'@' }, | ||||
| 	{  L("BS"),			L('\010') }, | ||||
| 	{  L("backspace"),		L('\b') }, | ||||
| 	{  L("HT"),			L('\011') }, | ||||
| 	{  L("tab"),			L('\t') }, | ||||
| 	{  L("LF"),			L('\012') }, | ||||
| 	{  L("newline"),		L('\n') }, | ||||
| 	{  L("VT"),			L('\013') }, | ||||
| 	{  L("vertical-tab"),		L('\v') }, | ||||
| 	{  L("FF"),			L('\014') }, | ||||
| 	{  L("form-feed"),		L('\f') }, | ||||
| 	{  L("CR"),			L('\015') }, | ||||
| 	{  L("carriage-return"),	L('\r') }, | ||||
| 	{  L("SO"),			L('\016') }, | ||||
| 	{  L("SI"),			L('\017') }, | ||||
| 	{  L("DLE"),			L('\020') }, | ||||
| 	{  L("DC1"),			L('\021') }, | ||||
| 	{  L("DC2"),			L('\022') }, | ||||
| 	{  L("DC3"),			L('\023') }, | ||||
| 	{  L("DC4"),			L('\024') }, | ||||
| 	{  L("NAK"),			L('\025') }, | ||||
| 	{  L("SYN"),			L('\026') }, | ||||
| 	{  L("ETB"),			L('\027') }, | ||||
| 	{  L("CAN"),			L('\030') }, | ||||
| 	{  L("EM"),			L('\031') }, | ||||
| 	{  L("SUB"),			L('\032') }, | ||||
| 	{  L("ESC"),			L('\033') }, | ||||
| 	{  L("IS4"),			L('\034') }, | ||||
| 	{  L("FS"),			L('\034') }, | ||||
| 	{  L("IS3"),			L('\035') }, | ||||
| 	{  L("GS"),			L('\035') }, | ||||
| 	{  L("IS2"),			L('\036') }, | ||||
| 	{  L("RS"),			L('\036') }, | ||||
| 	{  L("IS1"),			L('\037') }, | ||||
| 	{  L("US"),			L('\037') }, | ||||
| 	{  L("space"),			L(' ') }, | ||||
| 	{  L("exclamation-mark"),	L('!') }, | ||||
| 	{  L("quotation-mark"),		L('"') }, | ||||
| 	{  L("number-sign"),		L('#') }, | ||||
| 	{  L("dollar-sign"),		L('$') }, | ||||
| 	{  L("percent-sign"),		L('%') }, | ||||
| 	{  L("ampersand"),		L('&') }, | ||||
| 	{  L("apostrophe"),		L('\'') }, | ||||
| 	{  L("left-parenthesis"),	L('(') }, | ||||
| 	{  L("right-parenthesis"),	L(')') }, | ||||
| 	{  L("asterisk"),		L('*') }, | ||||
| 	{  L("plus-sign"),		L('+') }, | ||||
| 	{  L("comma"),			L(',') }, | ||||
| 	{  L("hyphen"),			L('-') }, | ||||
| 	{  L("hyphen-minus"),		L('-') }, | ||||
| 	{  L("minus"),			L('-') },	/* extension from POSIX.2 */ | ||||
| 	{  L("dash"),			L('-') },	/* extension from POSIX.2 */ | ||||
| 	{  L("period"),			L('.') }, | ||||
| 	{  L("full-stop"),		L('.') }, | ||||
| 	{  L("slash"),			L('/') }, | ||||
| 	{  L("solidus"),		L('/') },	/* extension from POSIX.2 */ | ||||
| 	{  L("zero"),			L('0') }, | ||||
| 	{  L("one"),			L('1') }, | ||||
| 	{  L("two"),			L('2') }, | ||||
| 	{  L("three"),			L('3') }, | ||||
| 	{  L("four"),			L('4') }, | ||||
| 	{  L("five"),			L('5') }, | ||||
| 	{  L("six"),			L('6') }, | ||||
| 	{  L("seven"),			L('7') }, | ||||
| 	{  L("eight"),			L('8') }, | ||||
| 	{  L("nine"),			L('9') }, | ||||
| 	{  L("colon"),			L(':') }, | ||||
| 	{  L("semicolon"),		L(';') }, | ||||
| 	{  L("less-than-sign"),		L('<') }, | ||||
| 	{  L("equals-sign"),		L('=') }, | ||||
| 	{  L("greater-than-sign"),	L('>') }, | ||||
| 	{  L("question-mark"),		L('?') }, | ||||
| 	{  L("commercial-at"),		L('@') }, | ||||
| 	/* upper-case letters omitted */ | ||||
| 	{  "left-square-bracket",'[' }, | ||||
| 	{  "backslash",		'\\' }, | ||||
| 	{  "reverse-solidus",	'\\' }, | ||||
| 	{  "right-square-bracket",']' }, | ||||
| 	{  "circumflex",	'^' }, | ||||
| 	{  "circumflex-accent",	'^' },		/* extension from POSIX.2 */ | ||||
| 	{  "underscore",	'_' }, | ||||
| 	{  "grave-accent",	'`' }, | ||||
| 	{  L("left-square-bracket"),	L('[') }, | ||||
| 	{  L("backslash"),		L('\\') }, | ||||
| 	{  L("reverse-solidus"),	L('\\') }, | ||||
| 	{  L("right-square-bracket"),	L(']') }, | ||||
| 	{  L("circumflex"),		L('^') }, | ||||
| 	{  L("circumflex-accent"),	L('^') },	/* extension from POSIX.2 */ | ||||
| 	{  L("underscore"),		L('_') }, | ||||
| 	{  L("grave-accent"),		L('`') }, | ||||
| 	/* lower-case letters omitted */ | ||||
| 	{  "left-brace",	'{' },		/* extension from POSIX.2 */ | ||||
| 	{  "left-curly-bracket",'{' }, | ||||
| 	{  "vertical-line",	'|' }, | ||||
| 	{  "right-brace",	'}' },	/* extension from POSIX.2 */ | ||||
| 	{  "right-curly-bracket",'}' }, | ||||
| 	{  "tilde",		'~' }, | ||||
| 	{  "DEL",		'\177' }, | ||||
| 	{  L("left-brace"),		L('{') },	/* extension from POSIX.2 */ | ||||
| 	{  L("left-curly-bracket"),	L('{') }, | ||||
| 	{  L("vertical-line"),		L('|') }, | ||||
| 	{  L("right-brace"),		L('}') },	/* extension from POSIX.2 */ | ||||
| 	{  L("right-curly-bracket"),	L('}') }, | ||||
| 	{  L("tilde"),			L('~') }, | ||||
| 	{  L("DEL"),			L('\177') }, | ||||
| 	{  0,	0 }, | ||||
| }; | ||||
| 
 | ||||
| #endif | ||||
| #undef _COLLSYM | ||||
| #undef __COLLSYM | ||||
| #undef POSIXCOLL | ||||
|  |  | |||
							
								
								
									
										433
									
								
								lib/glob/glob.c
									
										
									
									
									
								
							
							
						
						
									
										433
									
								
								lib/glob/glob.c
									
										
									
									
									
								
							|  | @ -1,5 +1,6 @@ | |||
| /* File-name wildcard pattern matching for GNU.
 | ||||
|    Copyright (C) 1985, 1988, 1989 Free Software Foundation, Inc. | ||||
| /* glob.c -- file-name wildcard pattern matching for Bash.
 | ||||
| 
 | ||||
|    Copyright (C) 1985-2002 Free Software Foundation, Inc. | ||||
| 
 | ||||
|    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 | ||||
|  | @ -25,81 +26,33 @@ | |||
|   #pragma alloca | ||||
| #endif /* _AIX && RISC6000 && !__GNUC__ */ | ||||
| 
 | ||||
| #if defined (SHELL) | ||||
| #  include "bashtypes.h" | ||||
| #else | ||||
| #  include <sys/types.h> | ||||
| #endif | ||||
| #include "bashtypes.h" | ||||
| 
 | ||||
| #if defined (HAVE_UNISTD_H) | ||||
| #  include <unistd.h> | ||||
| #endif | ||||
| 
 | ||||
| #if defined (SHELL) | ||||
| #  include "bashansi.h" | ||||
| #else | ||||
| #  if defined (HAVE_STDLIB_H) | ||||
| #    include <stdlib.h> | ||||
| #  endif | ||||
| #  if defined (HAVE_STRING_H) | ||||
| #    include <string.h> | ||||
| #  else /* !HAVE_STRING_H */ | ||||
| #    include <strings.h> | ||||
| #  endif /* !HAVE_STRING_H */ | ||||
| #endif | ||||
| 
 | ||||
| #if defined (HAVE_DIRENT_H) | ||||
| #  include <dirent.h> | ||||
| #  define D_NAMLEN(d) strlen ((d)->d_name) | ||||
| #else /* !HAVE_DIRENT_H */ | ||||
| #  define D_NAMLEN(d) ((d)->d_namlen) | ||||
| #  if defined (HAVE_SYS_NDIR_H) | ||||
| #    include <sys/ndir.h> | ||||
| #  endif | ||||
| #  if defined (HAVE_SYS_DIR_H) | ||||
| #    include <sys/dir.h> | ||||
| #  endif /* HAVE_SYS_DIR_H */ | ||||
| #  if defined (HAVE_NDIR_H) | ||||
| #    include <ndir.h> | ||||
| #  endif | ||||
| #  if !defined (dirent) | ||||
| #    define dirent direct | ||||
| #  endif | ||||
| #endif /* !HAVE_DIRENT_H */ | ||||
| 
 | ||||
| #if defined (_POSIX_SOURCE) && !defined (STRUCT_DIRENT_HAS_D_INO) || defined (BROKEN_DIRENT_D_INO) | ||||
| /* Posix does not require that the d_ino field be present, and some
 | ||||
|    systems do not provide it. */ | ||||
| #  define REAL_DIR_ENTRY(dp) 1 | ||||
| #else | ||||
| #  define REAL_DIR_ENTRY(dp) (dp->d_ino != 0) | ||||
| #endif /* _POSIX_SOURCE */ | ||||
| 
 | ||||
| #if !defined (HAVE_BCOPY) && !defined (bcopy) | ||||
| #  define bcopy(s, d, n) ((void) memcpy ((d), (s), (n))) | ||||
| #endif /* !HAVE_BCOPY && !bcopy */ | ||||
| 
 | ||||
| #if defined (SHELL) | ||||
| #  include "posixstat.h" | ||||
| #else /* !SHELL */ | ||||
| #  include <sys/stat.h> | ||||
| #endif /* !SHELL */ | ||||
| #include "bashansi.h" | ||||
| #include "posixdir.h" | ||||
| #include "posixstat.h" | ||||
| #include "shmbutil.h" | ||||
| #include "xmalloc.h" | ||||
| 
 | ||||
| #include "filecntl.h" | ||||
| #if !defined (F_OK) | ||||
| #  define F_OK 0 | ||||
| #endif | ||||
| 
 | ||||
| #if defined (SHELL) | ||||
| #  include "memalloc.h" | ||||
| #endif | ||||
| #include "stdc.h" | ||||
| #include "memalloc.h" | ||||
| #include "quit.h" | ||||
| 
 | ||||
| #include "glob.h" | ||||
| #include "strmatch.h" | ||||
| 
 | ||||
| #if !defined (HAVE_STDLIB_H) && !defined (SHELL) | ||||
| extern char *malloc (), *realloc (); | ||||
| extern void free (); | ||||
| #endif /* !HAVE_STDLIB_H */ | ||||
| #if !defined (HAVE_BCOPY) && !defined (bcopy) | ||||
| #  define bcopy(s, d, n) ((void) memcpy ((d), (s), (n))) | ||||
| #endif /* !HAVE_BCOPY && !bcopy */ | ||||
| 
 | ||||
| #if !defined (NULL) | ||||
| #  if defined (__STDC__) | ||||
|  | @ -109,13 +62,10 @@ extern void free (); | |||
| #  endif /* __STDC__ */ | ||||
| #endif /* !NULL */ | ||||
| 
 | ||||
| #if defined (SHELL) | ||||
| extern void throw_to_top_level (); | ||||
| extern int test_eaccess (); | ||||
| extern void throw_to_top_level __P((void)); | ||||
| extern int test_eaccess __P((char *, int)); | ||||
| 
 | ||||
| extern int interrupt_state; | ||||
| extern int extended_glob; | ||||
| #endif /* SHELL */ | ||||
| 
 | ||||
| /* Global variable which controls whether or not * matches .*.
 | ||||
|    Non-zero means don't match .*.  */ | ||||
|  | @ -128,51 +78,152 @@ int glob_ignore_case = 0; | |||
| /* Global variable to return to signify an error in globbing. */ | ||||
| char *glob_error_return; | ||||
| 
 | ||||
| /* Return nonzero if PATTERN has any special globbing chars in it.  */ | ||||
| /* Some forward declarations. */ | ||||
| static int skipname __P((char *, char *)); | ||||
| #if HANDLE_MULTIBYTE | ||||
| static int mbskipname __P((char *, char *)); | ||||
| #endif | ||||
| #if HANDLE_MULTIBYTE | ||||
| static void udequote_pathname __P((char *)); | ||||
| static void wdequote_pathname __P((char *)); | ||||
| #else | ||||
| #  define dequote_pathname udequote_pathname | ||||
| #endif | ||||
| static void dequote_pathname __P((char *)); | ||||
| static int glob_testdir __P((char *)); | ||||
| static char **glob_dir_to_array __P((char *, char **, int)); | ||||
| 
 | ||||
| /* Compile `glob_loop.c' for single-byte characters. */ | ||||
| #define CHAR	unsigned char | ||||
| #define INT	int | ||||
| #define L(CS)	CS | ||||
| #define INTERNAL_GLOB_PATTERN_P internal_glob_pattern_p | ||||
| #include "glob_loop.c" | ||||
| 
 | ||||
| /* Compile `glob_loop.c' again for multibyte characters. */ | ||||
| #if HANDLE_MULTIBYTE | ||||
| 
 | ||||
| #define CHAR	wchar_t | ||||
| #define INT	wint_t | ||||
| #define L(CS)	L##CS | ||||
| #define INTERNAL_GLOB_PATTERN_P internal_glob_wpattern_p | ||||
| #include "glob_loop.c" | ||||
| 
 | ||||
| #endif /* HANDLE_MULTIBYTE */ | ||||
| 
 | ||||
| /* And now a function that calls either the single-byte or multibyte version
 | ||||
|    of internal_glob_pattern_p. */ | ||||
| int | ||||
| glob_pattern_p (pattern) | ||||
|      const char *pattern; | ||||
| { | ||||
|   register const char *p; | ||||
|   register char c; | ||||
|   int bopen; | ||||
| #if HANDLE_MULTIBYTE | ||||
|   mbstate_t ps; | ||||
|   size_t n; | ||||
|   wchar_t *wpattern; | ||||
|   int r; | ||||
| 
 | ||||
|   p = pattern; | ||||
|   bopen = 0; | ||||
|   if (MB_CUR_MAX == 1) | ||||
|     return (internal_glob_pattern_p (pattern)); | ||||
| 
 | ||||
|   while ((c = *p++) != '\0') | ||||
|     switch (c) | ||||
|       { | ||||
|       case '?': | ||||
|       case '*': | ||||
| 	return (1); | ||||
| 
 | ||||
|       case '[':		/* Only accept an open brace if there is a close */ | ||||
| 	bopen++;	/* brace to match it.  Bracket expressions must be */ | ||||
| 	continue;	/* complete, according to Posix.2 */ | ||||
|       case ']': | ||||
| 	if (bopen) | ||||
| 	  return (1); | ||||
| 	continue;       | ||||
| 
 | ||||
|       case '+':		/* extended matching operators */ | ||||
|       case '@': | ||||
|       case '!': | ||||
| 	if (*p == '(')	/*) */ | ||||
| 	  return (1); | ||||
| 	continue; | ||||
| 
 | ||||
|       case '\\': | ||||
| 	if (*p++ == '\0') | ||||
| 	  return (0); | ||||
|       } | ||||
| 
 | ||||
|   return (0); | ||||
|   /* Convert strings to wide chars, and call the multibyte version. */ | ||||
|   memset (&ps, '\0', sizeof (ps)); | ||||
|   n = xmbsrtowcs (NULL, (const char **)&pattern, 0, &ps); | ||||
|   if (n == (size_t)-1) | ||||
|     /* Oops.  Invalid multibyte sequence.  Try it as single-byte sequence. */ | ||||
|     return (internal_glob_pattern_p (pattern)); | ||||
|   wpattern = (wchar_t *)xmalloc ((n + 1) * sizeof (wchar_t)); | ||||
|   (void) xmbsrtowcs (wpattern, (const char **)&pattern, n + 1, &ps); | ||||
|   r = internal_glob_wpattern_p (wpattern); | ||||
|   free (wpattern); | ||||
|   return r; | ||||
| #else | ||||
|   return (internal_glob_pattern_p (pattern)); | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| /* Return 1 if DNAME should be skipped according to PAT.  Mostly concerned
 | ||||
|    with matching leading `.'. */ | ||||
| 
 | ||||
| static int | ||||
| skipname (pat, dname) | ||||
|      char *pat; | ||||
|      char *dname; | ||||
| { | ||||
|   /* If a leading dot need not be explicitly matched, and the pattern
 | ||||
|      doesn't start with a `.', don't match `.' or `..' */ | ||||
|   if (noglob_dot_filenames == 0 && pat[0] != '.' && | ||||
| 	(pat[0] != '\\' || pat[1] != '.') && | ||||
| 	(dname[0] == '.' && | ||||
| 	  (dname[1] == '\0' || (dname[1] == '.' && dname[2] == '\0')))) | ||||
|     return 1; | ||||
| 
 | ||||
|   /* If a dot must be explicity matched, check to see if they do. */ | ||||
|   else if (noglob_dot_filenames && dname[0] == '.' && pat[0] != '.' && | ||||
| 	(pat[0] != '\\' || pat[1] != '.')) | ||||
|     return 1; | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| #if HANDLE_MULTIBYTE | ||||
| /* Return 1 if DNAME should be skipped according to PAT.  Handles multibyte
 | ||||
|    characters in PAT and DNAME.  Mostly concerned with matching leading `.'. */ | ||||
| 
 | ||||
| static int | ||||
| mbskipname (pat, dname) | ||||
|      char *pat, *dname; | ||||
| { | ||||
|   char *pat_bak, *dn_bak; | ||||
|   wchar_t *pat_wc, *dn_wc; | ||||
|   mbstate_t pat_ps, dn_ps; | ||||
|   size_t pat_n, dn_n, n; | ||||
| 
 | ||||
|   n = strlen(pat); | ||||
|   pat_bak = (char *) alloca (n + 1); | ||||
|   memcpy (pat_bak, pat, n + 1); | ||||
| 
 | ||||
|   n = strlen(dname); | ||||
|   dn_bak = (char *) alloca (n + 1); | ||||
|   memcpy (dn_bak, dname,  n + 1); | ||||
| 
 | ||||
|   memset(&pat_ps, '\0', sizeof(mbstate_t)); | ||||
|   memset(&dn_ps, '\0', sizeof(mbstate_t)); | ||||
| 
 | ||||
|   pat_n = xmbsrtowcs (NULL, (const char **)&pat_bak, 0, &pat_ps); | ||||
|   dn_n = xmbsrtowcs (NULL, (const char **)&dn_bak, 0, &dn_ps); | ||||
| 
 | ||||
|   if (pat_n != (size_t)-1 && dn_n !=(size_t)-1) | ||||
|     { | ||||
|       pat_wc = (wchar_t *) alloca ((pat_n + 1) * sizeof(wchar_t)); | ||||
|       dn_wc = (wchar_t *) alloca ((dn_n + 1) * sizeof(wchar_t)); | ||||
| 
 | ||||
|       (void) xmbsrtowcs (pat_wc, (const char **)&pat_bak, pat_n + 1, &pat_ps); | ||||
|       (void) xmbsrtowcs (dn_wc, (const char **)&dn_bak, dn_n + 1, &dn_ps); | ||||
| 
 | ||||
|       /* If a leading dot need not be explicitly matched, and the
 | ||||
| 	 pattern doesn't start with a `.', don't match `.' or `..' */ | ||||
|       if (noglob_dot_filenames == 0 && pat_wc[0] != L'.' && | ||||
| 	    (pat_wc[0] != L'\\' || pat_wc[1] != L'.') && | ||||
| 	    (dn_wc[0] == L'.' && | ||||
| 	      (dn_wc[1] == L'\0' || (dn_wc[1] == L'.' && dn_wc[2] == L'\0')))) | ||||
| 	return 1; | ||||
| 
 | ||||
|       /* If a leading dot must be explicity matched, check to see if the
 | ||||
| 	 pattern and dirname both have one. */ | ||||
|      else if (noglob_dot_filenames && dn_wc[0] == L'.' && | ||||
| 	   pat_wc[0] != L'.' && | ||||
| 	   (pat_wc[0] != L'\\' || pat_wc[1] != L'.')) | ||||
| 	return 1; | ||||
|     } | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| #endif /* HANDLE_MULTIBYTE */ | ||||
| 
 | ||||
| /* Remove backslashes quoting characters in PATHNAME by modifying PATHNAME. */ | ||||
| static void | ||||
| dequote_pathname (pathname) | ||||
| udequote_pathname (pathname) | ||||
|      char *pathname; | ||||
| { | ||||
|   register int i, j; | ||||
|  | @ -190,18 +241,71 @@ dequote_pathname (pathname) | |||
|   pathname[j] = '\0'; | ||||
| } | ||||
| 
 | ||||
|  | ||||
| #if HANDLE_MULTIBYTE | ||||
| /* Remove backslashes quoting characters in PATHNAME by modifying PATHNAME. */ | ||||
| static void | ||||
| wdequote_pathname (pathname) | ||||
|      char *pathname; | ||||
| { | ||||
|   mbstate_t ps; | ||||
|   size_t len, n; | ||||
|   wchar_t *wpathname; | ||||
|   char *pathname_bak; | ||||
|   int i, j; | ||||
| 
 | ||||
|   len = strlen (pathname); | ||||
|   pathname_bak = (char *) alloca (len + 1); | ||||
|   memcpy (pathname_bak, pathname , len + 1); | ||||
| 
 | ||||
|   /* Convert the strings into wide characters.  */ | ||||
|   memset (&ps, '\0', sizeof (ps)); | ||||
|   n = xmbsrtowcs (NULL, (const char **)&pathname_bak, 0, &ps); | ||||
|   if (n == (size_t) -1) | ||||
|     /* Something wrong. */ | ||||
|     return; | ||||
| 
 | ||||
|   wpathname = (wchar_t *) alloca ((n + 1) * sizeof (wchar_t)); | ||||
|   (void) xmbsrtowcs (wpathname, (const char **)&pathname_bak, n + 1, &ps); | ||||
| 
 | ||||
|   for (i = j = 0; wpathname && wpathname[i]; ) | ||||
|     { | ||||
|       if (wpathname[i] == L'\\') | ||||
| 	i++; | ||||
| 
 | ||||
|       wpathname[j++] = wpathname[i++]; | ||||
| 
 | ||||
|       if (!wpathname[i - 1]) | ||||
| 	break; | ||||
|     } | ||||
|   wpathname[j] = L'\0'; | ||||
| 
 | ||||
|   /* Convert the wide character string into unibyte character set. */ | ||||
|   memset (&ps, '\0', sizeof(mbstate_t)); | ||||
|   n = wcsrtombs(pathname, (const wchar_t **)&wpathname, len, &ps); | ||||
|   pathname[len] = '\0'; | ||||
| } | ||||
| 
 | ||||
| static void | ||||
| dequote_pathname (pathname) | ||||
|      char *pathname; | ||||
| { | ||||
|   if (MB_CUR_MAX > 1) | ||||
|     wdequote_pathname (pathname); | ||||
|   else | ||||
|     udequote_pathname (pathname); | ||||
| } | ||||
| #endif /* HANDLE_MULTIBYTE */ | ||||
| 
 | ||||
| /* Test whether NAME exists. */ | ||||
| 
 | ||||
| #if defined (HAVE_LSTAT) | ||||
| #  define GLOB_TESTNAME(name)  (lstat (name, &finfo)) | ||||
| #else /* !HAVE_LSTAT */ | ||||
| #  if defined (SHELL) && !defined (AFS) | ||||
| #  if !defined (AFS) | ||||
| #    define GLOB_TESTNAME(name)  (test_eaccess (nextname, F_OK)) | ||||
| #  else /* !SHELL || AFS */ | ||||
| #  else /* AFS */ | ||||
| #    define GLOB_TESTNAME(name)  (access (nextname, F_OK)) | ||||
| #  endif /* !SHELL || AFS */ | ||||
| #  endif /* AFS */ | ||||
| #endif /* !HAVE_LSTAT */ | ||||
| 
 | ||||
| /* Return 0 if DIR is a directory, -1 otherwise. */ | ||||
|  | @ -237,9 +341,10 @@ glob_testdir (dir) | |||
|    Look in errno for more information.  */ | ||||
| 
 | ||||
| char ** | ||||
| glob_vector (pat, dir) | ||||
| glob_vector (pat, dir, flags) | ||||
|      char *pat; | ||||
|      char *dir; | ||||
|      int flags; | ||||
| { | ||||
|   struct globval | ||||
|     { | ||||
|  | @ -256,7 +361,7 @@ glob_vector (pat, dir) | |||
|   int lose, skip; | ||||
|   register char **name_vector; | ||||
|   register unsigned int i; | ||||
|   int flags;		/* Flags passed to strmatch (). */ | ||||
|   int mflags;		/* Flags passed to strmatch (). */ | ||||
| 
 | ||||
|   lastlink = 0; | ||||
|   count = lose = skip = 0; | ||||
|  | @ -344,17 +449,15 @@ glob_vector (pat, dir) | |||
| 
 | ||||
|       /* Compute the flags that will be passed to strmatch().  We don't
 | ||||
| 	 need to do this every time through the loop. */ | ||||
|       flags = (noglob_dot_filenames ? FNM_PERIOD : 0) | FNM_PATHNAME; | ||||
|       mflags = (noglob_dot_filenames ? FNM_PERIOD : 0) | FNM_PATHNAME; | ||||
| 
 | ||||
| #ifdef FNM_CASEFOLD | ||||
|       if (glob_ignore_case) | ||||
| 	flags |= FNM_CASEFOLD; | ||||
| 	mflags |= FNM_CASEFOLD; | ||||
| #endif | ||||
| 
 | ||||
| #ifdef SHELL | ||||
|       if (extended_glob) | ||||
| 	flags |= FNM_EXTMATCH; | ||||
| #endif | ||||
| 	mflags |= FNM_EXTMATCH; | ||||
| 
 | ||||
|       /* Scan the directory, finding all names that match.
 | ||||
| 	 For each name that matches, allocate a struct globval | ||||
|  | @ -362,14 +465,12 @@ glob_vector (pat, dir) | |||
| 	 Chain those structs together; lastlink is the front of the chain.  */ | ||||
|       while (1) | ||||
| 	{ | ||||
| #if defined (SHELL) | ||||
| 	  /* Make globbing interruptible in the shell. */ | ||||
| 	  if (interrupt_state) | ||||
| 	    { | ||||
| 	      lose = 1; | ||||
| 	      break; | ||||
| 	    } | ||||
| #endif /* SHELL */ | ||||
| 	   | ||||
| 	  dp = readdir (d); | ||||
| 	  if (dp == NULL) | ||||
|  | @ -379,22 +480,15 @@ glob_vector (pat, dir) | |||
| 	  if (REAL_DIR_ENTRY (dp) == 0) | ||||
| 	    continue; | ||||
| 
 | ||||
| 	  /* If a leading dot need not be explicitly matched, and the pattern
 | ||||
| 	     doesn't start with a `.', don't match `.' or `..' */ | ||||
| #define dname dp->d_name | ||||
| 	  if (noglob_dot_filenames == 0 && pat[0] != '.' && | ||||
| 		(pat[0] != '\\' || pat[1] != '.') && | ||||
| 		(dname[0] == '.' && | ||||
| 		  (dname[1] == '\0' || (dname[1] == '.' && dname[2] == '\0')))) | ||||
| #undef dname | ||||
| #if HANDLE_MULTIBYTE | ||||
| 	  if (MB_CUR_MAX > 1 && mbskipname (pat, dp->d_name)) | ||||
| 	    continue; | ||||
| 	  else | ||||
| #endif | ||||
| 	  if (skipname (pat, dp->d_name)) | ||||
| 	    continue; | ||||
| 
 | ||||
| 	  /* If a dot must be explicity matched, check to see if they do. */ | ||||
| 	  if (noglob_dot_filenames && dp->d_name[0] == '.' && pat[0] != '.' && | ||||
| 		(pat[0] != '\\' || pat[1] != '.')) | ||||
| 	    continue; | ||||
| 
 | ||||
| 	  if (strmatch (pat, dp->d_name, flags) != FNM_NOMATCH) | ||||
| 	  if (strmatch (pat, dp->d_name, mflags) != FNM_NOMATCH) | ||||
| 	    { | ||||
| 	      nextlink = (struct globval *) alloca (sizeof (struct globval)); | ||||
| 	      nextlink->next = lastlink; | ||||
|  | @ -429,10 +523,8 @@ glob_vector (pat, dir) | |||
| 	  free (lastlink->name); | ||||
| 	  lastlink = lastlink->next; | ||||
| 	} | ||||
| #if defined (SHELL) | ||||
|       if (interrupt_state) | ||||
| 	throw_to_top_level (); | ||||
| #endif /* SHELL */ | ||||
| 
 | ||||
|       QUIT; | ||||
| 
 | ||||
|       return ((char **)NULL); | ||||
|     } | ||||
|  | @ -447,22 +539,40 @@ glob_vector (pat, dir) | |||
|   name_vector[count] = NULL; | ||||
|   return (name_vector); | ||||
| } | ||||
|  | ||||
| 
 | ||||
| /* Return a new array which is the concatenation of each string in ARRAY
 | ||||
|    to DIR.  This function expects you to pass in an allocated ARRAY, and | ||||
|    it takes care of free()ing that array.  Thus, you might think of this | ||||
|    function as side-effecting ARRAY. */ | ||||
|    function as side-effecting ARRAY.  This should handle GX_MARKDIRS. */ | ||||
| static char ** | ||||
| glob_dir_to_array (dir, array) | ||||
| glob_dir_to_array (dir, array, flags) | ||||
|      char *dir, **array; | ||||
|      int flags; | ||||
| { | ||||
|   register unsigned int i, l; | ||||
|   int add_slash; | ||||
|   char **result; | ||||
|   char **result, *new; | ||||
|   struct stat sb; | ||||
| 
 | ||||
|   l = strlen (dir); | ||||
|   if (l == 0) | ||||
|     return (array); | ||||
|     { | ||||
|       if (flags & GX_MARKDIRS) | ||||
| 	for (i = 0; array[i]; i++) | ||||
| 	  { | ||||
| 	    if ((stat (array[i], &sb) == 0) && S_ISDIR (sb.st_mode)) | ||||
| 	      { | ||||
| 		l = strlen (array[i]); | ||||
| 		new = (char *)realloc (array[i], l + 2); | ||||
| 		if (new == 0) | ||||
| 		  return NULL; | ||||
| 		new[l] = '/'; | ||||
| 		new[l+1] = '\0'; | ||||
| 		array[i] = new; | ||||
| 	      } | ||||
| 	  } | ||||
|       return (array); | ||||
|     } | ||||
| 
 | ||||
|   add_slash = dir[l - 1] != '/'; | ||||
| 
 | ||||
|  | @ -476,8 +586,9 @@ glob_dir_to_array (dir, array) | |||
| 
 | ||||
|   for (i = 0; array[i] != NULL; i++) | ||||
|     { | ||||
|       result[i] = (char *) malloc (l + (add_slash ? 1 : 0) | ||||
| 				   + strlen (array[i]) + 1); | ||||
|       /* 3 == 1 for NUL, 1 for slash at end of DIR, 1 for GX_MARKDIRS */ | ||||
|       result[i] = (char *) malloc (l + strlen (array[i]) + 3); | ||||
| 
 | ||||
|       if (result[i] == NULL) | ||||
| 	return (NULL); | ||||
| 
 | ||||
|  | @ -485,6 +596,16 @@ glob_dir_to_array (dir, array) | |||
|       if (add_slash) | ||||
| 	result[i][l] = '/'; | ||||
|       strcpy (result[i] + l + add_slash, array[i]); | ||||
|       if (flags & GX_MARKDIRS) | ||||
| 	{ | ||||
| 	  if ((stat (result[i], &sb) == 0) && S_ISDIR (sb.st_mode)) | ||||
| 	    { | ||||
| 	      size_t rlen; | ||||
| 	      rlen = strlen (result[i]); | ||||
| 	      result[i][rlen] = '/'; | ||||
| 	      result[i][rlen+1] = '\0'; | ||||
| 	    } | ||||
| 	} | ||||
|     } | ||||
|   result[i] = NULL; | ||||
| 
 | ||||
|  | @ -495,15 +616,16 @@ glob_dir_to_array (dir, array) | |||
| 
 | ||||
|   return (result); | ||||
| } | ||||
|  | ||||
| 
 | ||||
| /* Do globbing on PATHNAME.  Return an array of pathnames that match,
 | ||||
|    marking the end of the array with a null-pointer as an element. | ||||
|    If no pathnames match, then the array is empty (first element is null). | ||||
|    If there isn't enough memory, then return NULL. | ||||
|    If a file system error occurs, return -1; `errno' has the error code.  */ | ||||
| char ** | ||||
| glob_filename (pathname) | ||||
| glob_filename (pathname, flags) | ||||
|      char *pathname; | ||||
|      int flags; | ||||
| { | ||||
|   char **result; | ||||
|   unsigned int result_size; | ||||
|  | @ -545,7 +667,7 @@ glob_filename (pathname) | |||
|       if (directory_name[directory_len - 1] == '/') | ||||
| 	directory_name[directory_len - 1] = '\0'; | ||||
| 
 | ||||
|       directories = glob_filename (directory_name); | ||||
|       directories = glob_filename (directory_name, flags & ~GX_MARKDIRS); | ||||
| 
 | ||||
|       if (directories == NULL) | ||||
| 	goto memory_error; | ||||
|  | @ -571,7 +693,7 @@ glob_filename (pathname) | |||
| 	  /* Scan directory even on a NULL pathname.  That way, `*h/'
 | ||||
| 	     returns only directories ending in `h', instead of all | ||||
| 	     files ending in `h' with a `/' appended. */ | ||||
| 	  temp_results = glob_vector (filename, directories[i]); | ||||
| 	  temp_results = glob_vector (filename, directories[i], flags & ~GX_MARKDIRS); | ||||
| 
 | ||||
| 	  /* Handle error cases. */ | ||||
| 	  if (temp_results == NULL) | ||||
|  | @ -584,7 +706,7 @@ glob_filename (pathname) | |||
| 	      char **array; | ||||
| 	      register unsigned int l; | ||||
| 
 | ||||
| 	      array = glob_dir_to_array (directories[i], temp_results); | ||||
| 	      array = glob_dir_to_array (directories[i], temp_results, flags); | ||||
| 	      l = 0; | ||||
| 	      while (array[l] != NULL) | ||||
| 		++l; | ||||
|  | @ -619,6 +741,7 @@ glob_filename (pathname) | |||
|       result = (char **) realloc ((char *) result, 2 * sizeof (char *)); | ||||
|       if (result == NULL) | ||||
| 	return (NULL); | ||||
|       /* Handle GX_MARKDIRS here. */ | ||||
|       result[0] = (char *) malloc (directory_len + 1); | ||||
|       if (result[0] == NULL) | ||||
| 	goto memory_error; | ||||
|  | @ -642,13 +765,14 @@ glob_filename (pathname) | |||
| 
 | ||||
|       /* Just return what glob_vector () returns appended to the
 | ||||
| 	 directory name. */ | ||||
|       temp_results = | ||||
| 	glob_vector (filename, (directory_len == 0 ? "." : directory_name)); | ||||
|       temp_results = glob_vector (filename, | ||||
| 				  (directory_len == 0 ? "." : directory_name), | ||||
| 				  flags & ~GX_MARKDIRS); | ||||
| 
 | ||||
|       if (temp_results == NULL || temp_results == (char **)&glob_error_return) | ||||
| 	return (temp_results); | ||||
| 
 | ||||
|       return (glob_dir_to_array (directory_name, temp_results)); | ||||
|       return (glob_dir_to_array (directory_name, temp_results, flags)); | ||||
|     } | ||||
| 
 | ||||
|   /* We get to memory_error if the program has run out of memory, or
 | ||||
|  | @ -661,13 +785,12 @@ glob_filename (pathname) | |||
| 	free (result[i]); | ||||
|       free ((char *) result); | ||||
|     } | ||||
| #if defined (SHELL) | ||||
|   if (interrupt_state) | ||||
|     throw_to_top_level (); | ||||
| #endif /* SHELL */ | ||||
| 
 | ||||
|   QUIT; | ||||
| 
 | ||||
|   return (NULL); | ||||
| } | ||||
|  | ||||
| 
 | ||||
| #if defined (TEST) | ||||
| 
 | ||||
| main (argc, argv) | ||||
|  | @ -678,7 +801,7 @@ main (argc, argv) | |||
| 
 | ||||
|   for (i = 1; i < argc; ++i) | ||||
|     { | ||||
|       char **value = glob_filename (argv[i]); | ||||
|       char **value = glob_filename (argv[i], 0); | ||||
|       if (value == NULL) | ||||
| 	puts ("Out of memory."); | ||||
|       else if (value == &glob_error_return) | ||||
|  |  | |||
|  | @ -20,9 +20,13 @@ | |||
| 
 | ||||
| #include "stdc.h" | ||||
| 
 | ||||
| #define GX_MARKDIRS	0x01	/* mark directory names with trailing `/' */ | ||||
| #define GX_NOCASE	0x02	/* ignore case */ | ||||
| #define GX_MATCHDOT	0x04	/* match `.' literally */ | ||||
| 
 | ||||
| extern int glob_pattern_p __P((const char *)); | ||||
| extern char **glob_vector __P((char *, char *)); | ||||
| extern char **glob_filename __P((char *)); | ||||
| extern char **glob_vector __P((char *, char *, int)); | ||||
| extern char **glob_filename __P((char *, int)); | ||||
| 
 | ||||
| extern char *glob_error_return; | ||||
| extern int noglob_dot_filenames; | ||||
|  |  | |||
							
								
								
									
										67
									
								
								lib/glob/glob_loop.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								lib/glob/glob_loop.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,67 @@ | |||
| /* Copyright (C) 1991-2002 Free Software Foundation, Inc.
 | ||||
| 
 | ||||
|    This file is part of GNU Bash, the Bourne Again SHell. | ||||
|     | ||||
|    Bash 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, or (at your option) any later | ||||
|    version. | ||||
| 	       | ||||
|    Bash 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 Bash; see the file COPYING.  If not, write to the Free Software | ||||
|    Foundation, 59 Temple Place, Suite 330, Boston, MA 02111 USA. */ | ||||
| 
 | ||||
| static int INTERNAL_GLOB_PATTERN_P __P((const CHAR *)); | ||||
| 
 | ||||
| /* Return nonzero if PATTERN has any special globbing chars in it.
 | ||||
|    Compiled twice, once each for single-byte and multibyte characters. */ | ||||
| static int | ||||
| INTERNAL_GLOB_PATTERN_P (pattern) | ||||
|      const CHAR *pattern; | ||||
| { | ||||
|   register const CHAR *p; | ||||
|   register CHAR c; | ||||
|   int bopen; | ||||
| 
 | ||||
|   p = pattern; | ||||
|   bopen = 0; | ||||
| 
 | ||||
|   while ((c = *p++) != L('\0')) | ||||
|     switch (c) | ||||
|       { | ||||
|       case L('?'): | ||||
|       case L('*'): | ||||
| 	return 1; | ||||
| 
 | ||||
|       case L('['):      /* Only accept an open brace if there is a close */ | ||||
| 	bopen++;        /* brace to match it.  Bracket expressions must be */ | ||||
| 	continue;       /* complete, according to Posix.2 */ | ||||
|       case L(']'): | ||||
| 	if (bopen) | ||||
| 	  return 1; | ||||
| 	continue; | ||||
| 
 | ||||
|       case L('+'):         /* extended matching operators */ | ||||
|       case L('@'): | ||||
|       case L('!'): | ||||
| 	if (*p == L('('))  /*) */ | ||||
| 	  return 1; | ||||
| 	continue; | ||||
| 
 | ||||
|       case L('\\'): | ||||
| 	if (*p++ == L('\0')) | ||||
| 	  return 0; | ||||
|       } | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| #undef INTERNAL_GLOB_PATTERN_P | ||||
| #undef L | ||||
| #undef INT | ||||
| #undef CHAR | ||||
							
								
								
									
										737
									
								
								lib/glob/sm_loop.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										737
									
								
								lib/glob/sm_loop.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,737 @@ | |||
| /* Copyright (C) 1991-2002 Free Software Foundation, Inc.
 | ||||
| 
 | ||||
|    This file is part of GNU Bash, the Bourne Again SHell. | ||||
|     | ||||
|    Bash 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, or (at your option) any later | ||||
|    version. | ||||
| 	       | ||||
|    Bash 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 Bash; see the file COPYING.  If not, write to the Free Software | ||||
|    Foundation, 59 Temple Place, Suite 330, Boston, MA 02111 USA. */ | ||||
| 
 | ||||
| static int FCT __P((CHAR *, CHAR *, int)); | ||||
| static int GMATCH __P((CHAR *, CHAR *, CHAR *, CHAR *, int)); | ||||
| static CHAR *PARSE_COLLSYM __P((CHAR *, INT *)); | ||||
| static CHAR *BRACKMATCH __P((CHAR *, U_CHAR, int)); | ||||
| static int EXTMATCH __P((INT, CHAR *, CHAR *, CHAR *, CHAR *, int)); | ||||
| static CHAR *PATSCAN __P((CHAR *, CHAR *, INT)); | ||||
| 
 | ||||
| static int | ||||
| FCT (pattern, string, flags) | ||||
|      CHAR *pattern; | ||||
|      CHAR *string; | ||||
|      int flags; | ||||
| { | ||||
|   CHAR *se, *pe; | ||||
| 
 | ||||
|   if (string == 0 || pattern == 0) | ||||
|     return FNM_NOMATCH; | ||||
| 
 | ||||
|   se = string + STRLEN ((XCHAR *)string); | ||||
|   pe = pattern + STRLEN ((XCHAR *)pattern); | ||||
| 
 | ||||
|   return (GMATCH (string, se, pattern, pe, flags)); | ||||
| } | ||||
| 
 | ||||
| /* Match STRING against the filename pattern PATTERN, returning zero if
 | ||||
|    it matches, FNM_NOMATCH if not.  */ | ||||
| static int | ||||
| GMATCH (string, se, pattern, pe, flags) | ||||
|      CHAR *string, *se; | ||||
|      CHAR *pattern, *pe; | ||||
|      int flags; | ||||
| { | ||||
|   CHAR *p, *n;		/* pattern, string */ | ||||
|   INT c;		/* current pattern character - XXX U_CHAR? */ | ||||
|   INT sc;		/* current string character - XXX U_CHAR? */ | ||||
| 
 | ||||
|   p = pattern; | ||||
|   n = string; | ||||
| 
 | ||||
|   if (string == 0 || pattern == 0) | ||||
|     return FNM_NOMATCH; | ||||
| 
 | ||||
| #if DEBUG_MATCHING | ||||
| fprintf(stderr, "gmatch: string = %s; se = %s\n", string, se); | ||||
| fprintf(stderr, "gmatch: pattern = %s; pe = %s\n", pattern, pe); | ||||
| #endif | ||||
| 
 | ||||
|   while (p < pe) | ||||
|     { | ||||
|       c = *p++; | ||||
|       c = FOLD (c); | ||||
| 
 | ||||
|       sc = n < se ? *n : '\0'; | ||||
| 
 | ||||
| #ifdef EXTENDED_GLOB | ||||
|       /* EXTMATCH () will handle recursively calling GMATCH, so we can
 | ||||
| 	 just return what EXTMATCH() returns. */ | ||||
|       if ((flags & FNM_EXTMATCH) && *p == L('(') && | ||||
| 	  (c == L('+') || c == L('*') || c == L('?') || c == L('@') || c == L('!'))) /* ) */ | ||||
| 	{ | ||||
| 	  int lflags; | ||||
| 	  /* If we're not matching the start of the string, we're not
 | ||||
| 	     concerned about the special cases for matching `.' */ | ||||
| 	  lflags = (n == string) ? flags : (flags & ~FNM_PERIOD); | ||||
| 	  return (EXTMATCH (c, n, se, p, pe, lflags)); | ||||
| 	} | ||||
| #endif /* EXTENDED_GLOB */ | ||||
| 
 | ||||
|       switch (c) | ||||
| 	{ | ||||
| 	case L('?'):		/* Match single character */ | ||||
| 	  if (sc == '\0') | ||||
| 	    return FNM_NOMATCH; | ||||
| 	  else if ((flags & FNM_PATHNAME) && sc == L('/')) | ||||
| 	    /* If we are matching a pathname, `?' can never match a `/'. */ | ||||
| 	    return FNM_NOMATCH; | ||||
| 	  else if ((flags & FNM_PERIOD) && sc == L('.') && | ||||
| 		   (n == string || ((flags & FNM_PATHNAME) && n[-1] == L('/')))) | ||||
| 	    /* `?' cannot match a `.' if it is the first character of the
 | ||||
| 	       string or if it is the first character following a slash and | ||||
| 	       we are matching a pathname. */ | ||||
| 	    return FNM_NOMATCH; | ||||
| 	  break; | ||||
| 
 | ||||
| 	case L('\\'):		/* backslash escape removes special meaning */ | ||||
| 	  if (p == pe) | ||||
| 	    return FNM_NOMATCH; | ||||
| 
 | ||||
| 	  if ((flags & FNM_NOESCAPE) == 0) | ||||
| 	    { | ||||
| 	      c = *p++; | ||||
| 	      /* A trailing `\' cannot match. */ | ||||
| 	      if (p > pe) | ||||
| 		return FNM_NOMATCH; | ||||
| 	      c = FOLD (c); | ||||
| 	    } | ||||
| 	  if (FOLD (sc) != (U_CHAR)c) | ||||
| 	    return FNM_NOMATCH; | ||||
| 	  break; | ||||
| 
 | ||||
| 	case '*':		/* Match zero or more characters */ | ||||
| 	  if (p == pe) | ||||
| 	    return 0; | ||||
| 	   | ||||
| 	  if ((flags & FNM_PERIOD) && sc == L('.') && | ||||
| 	      (n == string || ((flags & FNM_PATHNAME) && n[-1] == L('/')))) | ||||
| 	    /* `*' cannot match a `.' if it is the first character of the
 | ||||
| 	       string or if it is the first character following a slash and | ||||
| 	       we are matching a pathname. */ | ||||
| 	    return FNM_NOMATCH; | ||||
| 
 | ||||
| 	  /* Collapse multiple consecutive `*' and `?', but make sure that
 | ||||
| 	     one character of the string is consumed for each `?'. */ | ||||
| 	  for (c = *p++; (c == L('?') || c == L('*')); c = *p++) | ||||
| 	    { | ||||
| 	      if ((flags & FNM_PATHNAME) && sc == L('/')) | ||||
| 		/* A slash does not match a wildcard under FNM_PATHNAME. */ | ||||
| 		return FNM_NOMATCH; | ||||
| 	      else if (c == L('?')) | ||||
| 		{ | ||||
| 		  if (sc == L('\0')) | ||||
| 		    return FNM_NOMATCH; | ||||
| 		  /* One character of the string is consumed in matching
 | ||||
| 		     this ? wildcard, so *??? won't match if there are | ||||
| 		     fewer than three characters. */ | ||||
| 		  n++; | ||||
| 		  sc = n < se ? *n : '\0'; | ||||
| 		} | ||||
| 
 | ||||
| #ifdef EXTENDED_GLOB | ||||
| 	      /* Handle ******(patlist) */ | ||||
| 	      if ((flags & FNM_EXTMATCH) && c == L('*') && *p == L('('))  /*)*/ | ||||
| 		{ | ||||
| 		  CHAR *newn; | ||||
| 		  /* We need to check whether or not the extended glob
 | ||||
| 		     pattern matches the remainder of the string. | ||||
| 		     If it does, we match the entire pattern. */ | ||||
| 		  for (newn = n; newn < se; ++newn) | ||||
| 		    { | ||||
| 		      if (EXTMATCH (c, newn, se, p, pe, flags) == 0) | ||||
| 			return (0); | ||||
| 		    } | ||||
| 		  /* We didn't match the extended glob pattern, but
 | ||||
| 		     that's OK, since we can match 0 or more occurrences. | ||||
| 		     We need to skip the glob pattern and see if we | ||||
| 		     match the rest of the string. */ | ||||
| 		  newn = PATSCAN (p + 1, pe, 0); | ||||
| 		  /* If NEWN is 0, we have an ill-formed pattern. */ | ||||
| 		  p = newn ? newn : pe; | ||||
| 		} | ||||
| #endif | ||||
| 	      if (p == pe) | ||||
| 		break; | ||||
| 	    } | ||||
| 
 | ||||
| 	  /* If we've hit the end of the pattern and the last character of
 | ||||
| 	     the pattern was handled by the loop above, we've succeeded. | ||||
| 	     Otherwise, we need to match that last character. */ | ||||
| 	  if (p == pe && (c == L('?') || c == L('*'))) | ||||
| 	    return (0); | ||||
| 
 | ||||
| 	  /* General case, use recursion. */ | ||||
| 	  { | ||||
| 	    U_CHAR c1; | ||||
| 
 | ||||
| 	    c1 = ((flags & FNM_NOESCAPE) == 0 && c == L('\\')) ? *p : c; | ||||
| 	    c1 = FOLD (c1); | ||||
| 	    for (--p; n < se; ++n) | ||||
| 	      { | ||||
| 		/* Only call strmatch if the first character indicates a
 | ||||
| 		   possible match.  We can check the first character if | ||||
| 		   we're not doing an extended glob match. */ | ||||
| 		if ((flags & FNM_EXTMATCH) == 0 && c != L('[') && FOLD (*n) != c1) /*]*/ | ||||
| 		  continue; | ||||
| 
 | ||||
| 		/* If we're doing an extended glob match and the pattern is not
 | ||||
| 		   one of the extended glob patterns, we can check the first | ||||
| 		   character. */ | ||||
| 		if ((flags & FNM_EXTMATCH) && p[1] != L('(') && /*)*/ | ||||
| 		    STRCHR (L("?*+@!"), *p) == 0 && c != L('[') && FOLD (*n) != c1) /*]*/ | ||||
| 		  continue; | ||||
| 
 | ||||
| 		/* Otherwise, we just recurse. */ | ||||
| 		if (GMATCH (n, se, p, pe, flags & ~FNM_PERIOD) == 0) | ||||
| 		  return (0); | ||||
| 	      } | ||||
| 	    return FNM_NOMATCH; | ||||
| 	  } | ||||
| 
 | ||||
| 	case L('['): | ||||
| 	  { | ||||
| 	    if (sc == L('\0') || n == se) | ||||
| 	      return FNM_NOMATCH; | ||||
| 
 | ||||
| 	    /* A character class cannot match a `.' if it is the first
 | ||||
| 	       character of the string or if it is the first character | ||||
| 	       following a slash and we are matching a pathname. */ | ||||
| 	    if ((flags & FNM_PERIOD) && sc == L('.') && | ||||
| 		(n == string || ((flags & FNM_PATHNAME) && n[-1] == L('/')))) | ||||
| 	      return (FNM_NOMATCH); | ||||
| 
 | ||||
| 	    p = BRACKMATCH (p, sc, flags); | ||||
| 	    if (p == 0) | ||||
| 	      return FNM_NOMATCH; | ||||
| 	  } | ||||
| 	  break; | ||||
| 
 | ||||
| 	default: | ||||
| 	  if ((U_CHAR)c != FOLD (sc)) | ||||
| 	    return (FNM_NOMATCH); | ||||
| 	} | ||||
| 
 | ||||
|       ++n; | ||||
|     } | ||||
| 
 | ||||
|   if (n == se) | ||||
|     return (0); | ||||
| 
 | ||||
|   if ((flags & FNM_LEADING_DIR) && *n == L('/')) | ||||
|     /* The FNM_LEADING_DIR flag says that "foo*" matches "foobar/frobozz".  */ | ||||
|     return 0; | ||||
| 	   | ||||
|   return (FNM_NOMATCH); | ||||
| } | ||||
| 
 | ||||
| /* Parse a bracket expression collating symbol ([.sym.]) starting at P, find
 | ||||
|    the value of the symbol, and move P past the collating symbol expression. | ||||
|    The value is returned in *VP, if VP is not null. */ | ||||
| static CHAR * | ||||
| PARSE_COLLSYM (p, vp) | ||||
|      CHAR *p; | ||||
|      INT *vp; | ||||
| { | ||||
|   register int pc; | ||||
|   INT val; | ||||
| 
 | ||||
|   p++;				/* move past the `.' */ | ||||
| 	   | ||||
|   for (pc = 0; p[pc]; pc++) | ||||
|     if (p[pc] == L('.') && p[pc+1] == L(']')) | ||||
|       break; | ||||
|    val = COLLSYM (p, pc); | ||||
|    if (vp) | ||||
|      *vp = val; | ||||
|    return (p + pc + 2); | ||||
| } | ||||
| 
 | ||||
| /* Use prototype definition here because of type promotion. */ | ||||
| static CHAR * | ||||
| #if defined (PROTOTYPES) | ||||
| BRACKMATCH (CHAR *p, U_CHAR test, int flags) | ||||
| #else | ||||
| BRACKMATCH (p, test, flags) | ||||
|      CHAR *p; | ||||
|      U_CHAR test; | ||||
|      int flags; | ||||
| #endif | ||||
| { | ||||
|   register CHAR cstart, cend, c; | ||||
|   register int not;    /* Nonzero if the sense of the character class is inverted.  */ | ||||
|   int brcnt; | ||||
|   INT pc; | ||||
|   CHAR *savep; | ||||
| 
 | ||||
|   test = FOLD (test); | ||||
| 
 | ||||
|   savep = p; | ||||
| 
 | ||||
|   /* POSIX.2 3.13.1 says that an exclamation mark (`!') shall replace the
 | ||||
|      circumflex (`^') in its role in a `nonmatching list'.  A bracket | ||||
|      expression starting with an unquoted circumflex character produces | ||||
|      unspecified results.  This implementation treats the two identically. */ | ||||
|   if (not = (*p == L('!') || *p == L('^'))) | ||||
|     ++p; | ||||
| 
 | ||||
|   c = *p++; | ||||
|   for (;;) | ||||
|     { | ||||
|       /* Initialize cstart and cend in case `-' is the last
 | ||||
| 	 character of the pattern. */ | ||||
|       cstart = cend = c; | ||||
| 
 | ||||
|       /* POSIX.2 equivalence class:  [=c=].  See POSIX.2 2.8.3.2.  Find
 | ||||
| 	 the end of the equivalence class, move the pattern pointer past | ||||
| 	 it, and check for equivalence.  XXX - this handles only | ||||
| 	 single-character equivalence classes, which is wrong, or at | ||||
| 	 least incomplete. */ | ||||
|       if (c == L('[') && *p == L('=') && p[2] == L('=') && p[3] == L(']')) | ||||
| 	{ | ||||
| 	  pc = FOLD (p[1]); | ||||
| 	  p += 4; | ||||
| 	  if (COLLEQUIV (test, pc)) | ||||
| 	    { | ||||
| /*[*/	      /* Move past the closing `]', since the first thing we do at
 | ||||
| 		 the `matched:' label is back p up one. */ | ||||
| 	      p++; | ||||
| 	      goto matched; | ||||
| 	    } | ||||
| 	  else | ||||
| 	    { | ||||
| 	      c = *p++; | ||||
| 	      if (c == L('\0')) | ||||
| 		return ((test == L('[')) ? savep : (CHAR *)0); /*]*/ | ||||
| 	      c = FOLD (c); | ||||
| 	      continue; | ||||
| 	    } | ||||
| 	} | ||||
| 
 | ||||
|       /* POSIX.2 character class expression.  See POSIX.2 2.8.3.2. */ | ||||
|       if (c == L('[') && *p == L(':')) | ||||
| 	{ | ||||
| 	  CHAR *close, *ccname; | ||||
| 
 | ||||
| 	  pc = 0;	/* make sure invalid char classes don't match. */ | ||||
| 	  /* Find end of character class name */ | ||||
| 	  for (close = p + 1; *close != '\0'; close++) | ||||
| 	    if (*close == L(':') && *(close+1) == L(']')) | ||||
| 	      break; | ||||
| 
 | ||||
| 	  if (*close != L('\0')) | ||||
| 	    { | ||||
| 	      ccname = (CHAR *)malloc ((close - p) * sizeof (CHAR)); | ||||
| 	      if (ccname == 0) | ||||
| 		pc = 0; | ||||
| 	      else | ||||
| 		{ | ||||
| 		  bcopy (p + 1, ccname, (close - p - 1) * sizeof (CHAR)); | ||||
| 		  *(ccname + (close - p - 1)) = L('\0'); | ||||
| 		  pc = IS_CCLASS (test, ccname); | ||||
| 		} | ||||
| 	      if (pc == -1) | ||||
| 		pc = 0; | ||||
| 	      else | ||||
| 		p = close + 2; | ||||
| 
 | ||||
| 	      free (ccname); | ||||
| 	    } | ||||
| 	     | ||||
| 	  if (pc) | ||||
| 	    { | ||||
| /*[*/	      /* Move past the closing `]', since the first thing we do at
 | ||||
| 		 the `matched:' label is back p up one. */ | ||||
| 	      p++; | ||||
| 	      goto matched; | ||||
| 	    } | ||||
| 	  else | ||||
| 	    { | ||||
| 	      /* continue the loop here, since this expression can't be
 | ||||
| 		 the first part of a range expression. */ | ||||
| 	      c = *p++; | ||||
| 	      if (c == L('\0')) | ||||
| 		return ((test == L('[')) ? savep : (CHAR *)0); | ||||
| 	      else if (c == L(']')) | ||||
| 		break; | ||||
| 	      c = FOLD (c); | ||||
| 	      continue; | ||||
| 	    } | ||||
| 	} | ||||
|   | ||||
|       /* POSIX.2 collating symbols.  See POSIX.2 2.8.3.2.  Find the end of
 | ||||
| 	 the symbol name, make sure it is terminated by `.]', translate | ||||
| 	 the name to a character using the external table, and do the | ||||
| 	 comparison. */ | ||||
|       if (c == L('[') && *p == L('.')) | ||||
| 	{ | ||||
| 	  p = PARSE_COLLSYM (p, &pc); | ||||
| 	  /* An invalid collating symbol cannot be the first point of a
 | ||||
| 	     range.  If it is, we set cstart to one greater than `test', | ||||
| 	     so any comparisons later will fail. */ | ||||
| 	  cstart = (pc == INVALID) ? test + 1 : pc; | ||||
| 	} | ||||
| 
 | ||||
|       if (!(flags & FNM_NOESCAPE) && c == L('\\')) | ||||
| 	{ | ||||
| 	  if (*p == '\0') | ||||
| 	    return (CHAR *)0; | ||||
| 	  cstart = cend = *p++; | ||||
| 	} | ||||
| 
 | ||||
|       cstart = cend = FOLD (cstart); | ||||
| 
 | ||||
|       /* POSIX.2 2.8.3.1.2 says: `An expression containing a `[' that
 | ||||
| 	 is not preceded by a backslash and is not part of a bracket | ||||
| 	 expression produces undefined results.'  This implementation | ||||
| 	 treats the `[' as just a character to be matched if there is | ||||
| 	 not a closing `]'. */ | ||||
|       if (c == L('\0')) | ||||
| 	return ((test == L('[')) ? savep : (CHAR *)0); | ||||
| 
 | ||||
|       c = *p++; | ||||
|       c = FOLD (c); | ||||
| 
 | ||||
|       if ((flags & FNM_PATHNAME) && c == L('/')) | ||||
| 	/* [/] can never match when matching a pathname.  */ | ||||
| 	return (CHAR *)0; | ||||
| 
 | ||||
|       /* This introduces a range, unless the `-' is the last
 | ||||
| 	 character of the class.  Find the end of the range | ||||
| 	 and move past it. */ | ||||
|       if (c == L('-') && *p != L(']')) | ||||
| 	{ | ||||
| 	  cend = *p++; | ||||
| 	  if (!(flags & FNM_NOESCAPE) && cend == L('\\')) | ||||
| 	    cend = *p++; | ||||
| 	  if (cend == L('\0')) | ||||
| 	    return (CHAR *)0; | ||||
| 	  if (cend == L('[') && *p == L('.')) | ||||
| 	    { | ||||
| 	      p = PARSE_COLLSYM (p, &pc); | ||||
| 	      /* An invalid collating symbol cannot be the second part of a
 | ||||
| 		 range expression.  If we get one, we set cend to one fewer | ||||
| 		 than the test character to make sure the range test fails. */ | ||||
| 	      cend = (pc == INVALID) ? test - 1 : pc; | ||||
| 	    } | ||||
| 	  cend = FOLD (cend); | ||||
| 
 | ||||
| 	  c = *p++; | ||||
| 
 | ||||
| 	  /* POSIX.2 2.8.3.2:  ``The ending range point shall collate
 | ||||
| 	     equal to or higher than the starting range point; otherwise | ||||
| 	     the expression shall be treated as invalid.''  Note that this | ||||
| 	     applies to only the range expression; the rest of the bracket | ||||
| 	     expression is still checked for matches. */ | ||||
| 	  if (RANGECMP (cstart, cend) > 0) | ||||
| 	    { | ||||
| 	      if (c == L(']')) | ||||
| 		break; | ||||
| 	      c = FOLD (c); | ||||
| 	      continue; | ||||
| 	    } | ||||
| 	} | ||||
| 
 | ||||
|       if (RANGECMP (test, cstart) >= 0 && RANGECMP (test, cend) <= 0) | ||||
| 	goto matched; | ||||
| 
 | ||||
|       if (c == L(']')) | ||||
| 	break; | ||||
|     } | ||||
|   /* No match. */ | ||||
|   return (!not ? (CHAR *)0 : p); | ||||
| 
 | ||||
| matched: | ||||
|   /* Skip the rest of the [...] that already matched.  */ | ||||
|   c = *--p; | ||||
|   brcnt = 1; | ||||
|   while (brcnt > 0) | ||||
|     { | ||||
|       /* A `[' without a matching `]' is just another character to match. */ | ||||
|       if (c == L('\0')) | ||||
| 	return ((test == L('[')) ? savep : (CHAR *)0); | ||||
| 
 | ||||
|       c = *p++; | ||||
|       if (c == L('[') && (*p == L('=') || *p == L(':') || *p == L('.'))) | ||||
| 	brcnt++; | ||||
|       else if (c == L(']')) | ||||
| 	brcnt--; | ||||
|       else if (!(flags & FNM_NOESCAPE) && c == L('\\')) | ||||
| 	{ | ||||
| 	  if (*p == '\0') | ||||
| 	    return (CHAR *)0; | ||||
| 	  /* XXX 1003.2d11 is unclear if this is right. */ | ||||
| 	  ++p; | ||||
| 	} | ||||
|     } | ||||
|   return (not ? (CHAR *)0 : p); | ||||
| } | ||||
| 
 | ||||
| #if defined (EXTENDED_GLOB) | ||||
| /* ksh-like extended pattern matching:
 | ||||
| 
 | ||||
| 	[?*+@!](pat-list) | ||||
| 
 | ||||
|    where pat-list is a list of one or patterns separated by `|'.  Operation | ||||
|    is as follows: | ||||
| 
 | ||||
| 	?(patlist)	match zero or one of the given patterns | ||||
| 	*(patlist)	match zero or more of the given patterns | ||||
| 	+(patlist)	match one or more of the given patterns | ||||
| 	@(patlist)	match exactly one of the given patterns | ||||
| 	!(patlist)	match anything except one of the given patterns | ||||
| */ | ||||
| 
 | ||||
| /* Scan a pattern starting at STRING and ending at END, keeping track of
 | ||||
|    embedded () and [].  If DELIM is 0, we scan until a matching `)' | ||||
|    because we're scanning a `patlist'.  Otherwise, we scan until we see | ||||
|    DELIM.  In all cases, we never scan past END.  The return value is the | ||||
|    first character after the matching DELIM. */ | ||||
| static CHAR * | ||||
| PATSCAN (string, end, delim) | ||||
|      CHAR *string, *end; | ||||
|      INT delim; | ||||
| { | ||||
|   int pnest, bnest; | ||||
|   INT cchar; | ||||
|   CHAR *s, c, *bfirst; | ||||
| 
 | ||||
|   pnest = bnest = 0; | ||||
|   cchar = 0; | ||||
|   bfirst = NULL; | ||||
| 
 | ||||
|   for (s = string; c = *s; s++) | ||||
|     { | ||||
|       if (s >= end) | ||||
| 	return (s); | ||||
|       switch (c) | ||||
| 	{ | ||||
| 	case L('\0'): | ||||
| 	  return ((CHAR *)NULL); | ||||
| 
 | ||||
| 	/* `[' is not special inside a bracket expression, but it may
 | ||||
| 	   introduce one of the special POSIX bracket expressions | ||||
| 	   ([.SYM.], [=c=], [: ... :]) that needs special handling. */ | ||||
| 	case L('['): | ||||
| 	  if (bnest == 0) | ||||
| 	    { | ||||
| 	      bfirst = s + 1; | ||||
| 	      if (*bfirst == L('!') || *bfirst == L('^')) | ||||
| 		bfirst++; | ||||
| 	      bnest++; | ||||
| 	    } | ||||
| 	  else if (s[1] == L(':') || s[1] == L('.') || s[1] == L('=')) | ||||
| 	    cchar = s[1]; | ||||
| 	  break; | ||||
| 
 | ||||
| 	/* `]' is not special if it's the first char (after a leading `!'
 | ||||
| 	   or `^') in a bracket expression or if it's part of one of the | ||||
| 	   special POSIX bracket expressions ([.SYM.], [=c=], [: ... :]) */ | ||||
| 	case L(']'): | ||||
| 	  if (bnest) | ||||
| 	    { | ||||
| 	      if (cchar && s[-1] == cchar) | ||||
| 		cchar = 0; | ||||
| 	      else if (s != bfirst) | ||||
| 		{ | ||||
| 		  bnest--; | ||||
| 		  bfirst = 0; | ||||
| 		} | ||||
| 	    } | ||||
| 	  break; | ||||
| 
 | ||||
| 	case L('('): | ||||
| 	  if (bnest == 0) | ||||
| 	    pnest++; | ||||
| 	  break; | ||||
| 
 | ||||
| 	case L(')'): | ||||
| 	  if (bnest == 0 && pnest-- <= 0) | ||||
| 	    return ++s; | ||||
| 	  break; | ||||
| 
 | ||||
| 	case L('|'): | ||||
| 	  if (bnest == 0 && pnest == 0 && delim == L('|')) | ||||
| 	    return ++s; | ||||
| 	  break; | ||||
| 	} | ||||
|     } | ||||
| 
 | ||||
|   return (NULL); | ||||
| } | ||||
| 
 | ||||
| /* Return 0 if dequoted pattern matches S in the current locale. */ | ||||
| static int | ||||
| STRCOMPARE (p, pe, s, se) | ||||
|      CHAR *p, *pe, *s, *se; | ||||
| { | ||||
|   int ret; | ||||
|   CHAR c1, c2; | ||||
| 
 | ||||
|   c1 = *pe; | ||||
|   c2 = *se; | ||||
| 
 | ||||
|   *pe = *se = '\0'; | ||||
| #if HAVE_MULTIBYTE || defined (HAVE_STRCOLL) | ||||
|   ret = STRCOLL ((XCHAR *)p, (XCHAR *)s); | ||||
| #else | ||||
|   ret = STRCMP ((XCHAR *)p, (XCHAR *)s); | ||||
| #endif | ||||
| 
 | ||||
|   *pe = c1; | ||||
|   *se = c2; | ||||
| 
 | ||||
|   return (ret == 0 ? ret : FNM_NOMATCH); | ||||
| } | ||||
| 
 | ||||
| /* Match a ksh extended pattern specifier.  Return FNM_NOMATCH on failure or
 | ||||
|    0 on success.  This is handed the entire rest of the pattern and string | ||||
|    the first time an extended pattern specifier is encountered, so it calls | ||||
|    gmatch recursively. */ | ||||
| static int | ||||
| EXTMATCH (xc, s, se, p, pe, flags) | ||||
|      INT xc;		/* select which operation */ | ||||
|      CHAR *s, *se; | ||||
|      CHAR *p, *pe; | ||||
|      int flags; | ||||
| { | ||||
|   CHAR *prest;			/* pointer to rest of pattern */ | ||||
|   CHAR *psub;			/* pointer to sub-pattern */ | ||||
|   CHAR *pnext;			/* pointer to next sub-pattern */ | ||||
|   CHAR *srest;			/* pointer to rest of string */ | ||||
|   int m1, m2; | ||||
| 
 | ||||
| #if DEBUG_MATCHING | ||||
| fprintf(stderr, "extmatch: xc = %c\n", xc); | ||||
| fprintf(stderr, "extmatch: s = %s; se = %s\n", s, se); | ||||
| fprintf(stderr, "extmatch: p = %s; pe = %s\n", p, pe); | ||||
| #endif | ||||
| 
 | ||||
|   prest = PATSCAN (p + (*p == L('(')), pe, 0); /* ) */ | ||||
|   if (prest == 0) | ||||
|     /* If PREST is 0, we failed to scan a valid pattern.  In this
 | ||||
|        case, we just want to compare the two as strings. */ | ||||
|     return (STRCOMPARE (p - 1, pe, s, se)); | ||||
| 
 | ||||
|   switch (xc) | ||||
|     { | ||||
|     case L('+'):		/* match one or more occurrences */ | ||||
|     case L('*'):		/* match zero or more occurrences */ | ||||
|       /* If we can get away with no matches, don't even bother.  Just
 | ||||
| 	 call GMATCH on the rest of the pattern and return success if | ||||
| 	 it succeeds. */ | ||||
|       if (xc == L('*') && (GMATCH (s, se, prest, pe, flags) == 0)) | ||||
| 	return 0; | ||||
| 
 | ||||
|       /* OK, we have to do this the hard way.  First, we make sure one of
 | ||||
| 	 the subpatterns matches, then we try to match the rest of the | ||||
| 	 string. */ | ||||
|       for (psub = p + 1; ; psub = pnext) | ||||
| 	{ | ||||
| 	  pnext = PATSCAN (psub, pe, L('|')); | ||||
| 	  for (srest = s; srest <= se; srest++) | ||||
| 	    { | ||||
| 	      /* Match this substring (S -> SREST) against this
 | ||||
| 		 subpattern (psub -> pnext - 1) */ | ||||
| 	      m1 = GMATCH (s, srest, psub, pnext - 1, flags) == 0; | ||||
| 	      /* OK, we matched a subpattern, so make sure the rest of the
 | ||||
| 		 string matches the rest of the pattern.  Also handle | ||||
| 		 multiple matches of the pattern. */ | ||||
| 	      if (m1) | ||||
| 		m2 = (GMATCH (srest, se, prest, pe, flags) == 0) || | ||||
| 		      (s != srest && GMATCH (srest, se, p - 1, pe, flags) == 0); | ||||
| 	      if (m1 && m2) | ||||
| 		return (0); | ||||
| 	    } | ||||
| 	  if (pnext == prest) | ||||
| 	    break; | ||||
| 	} | ||||
|       return (FNM_NOMATCH); | ||||
| 
 | ||||
|     case L('?'):		/* match zero or one of the patterns */ | ||||
|     case L('@'):		/* match exactly one of the patterns */ | ||||
|       /* If we can get away with no matches, don't even bother.  Just
 | ||||
| 	 call gmatch on the rest of the pattern and return success if | ||||
| 	 it succeeds. */ | ||||
|       if (xc == L('?') && (GMATCH (s, se, prest, pe, flags) == 0)) | ||||
| 	return 0; | ||||
| 
 | ||||
|       /* OK, we have to do this the hard way.  First, we see if one of
 | ||||
| 	 the subpatterns matches, then, if it does, we try to match the | ||||
| 	 rest of the string. */ | ||||
|       for (psub = p + 1; ; psub = pnext) | ||||
| 	{ | ||||
| 	  pnext = PATSCAN (psub, pe, L('|')); | ||||
| 	  srest = (prest == pe) ? se : s; | ||||
| 	  for ( ; srest <= se; srest++) | ||||
| 	    { | ||||
| 	      if (GMATCH (s, srest, psub, pnext - 1, flags) == 0 && | ||||
| 		  GMATCH (srest, se, prest, pe, flags) == 0) | ||||
| 		return (0); | ||||
| 	    } | ||||
| 	  if (pnext == prest) | ||||
| 	    break; | ||||
| 	} | ||||
|       return (FNM_NOMATCH); | ||||
| 
 | ||||
|     case '!':		/* match anything *except* one of the patterns */ | ||||
|       for (srest = s; srest <= se; srest++) | ||||
| 	{ | ||||
| 	  m1 = 0; | ||||
| 	  for (psub = p + 1; ; psub = pnext) | ||||
| 	    { | ||||
| 	      pnext = PATSCAN (psub, pe, L('|')); | ||||
| 	      /* If one of the patterns matches, just bail immediately. */ | ||||
| 	      if (m1 = (GMATCH (s, srest, psub, pnext - 1, flags) == 0)) | ||||
| 		break; | ||||
| 	      if (pnext == prest) | ||||
| 		break; | ||||
| 	    } | ||||
| 	  if (m1 == 0 && GMATCH (srest, se, prest, pe, flags) == 0) | ||||
| 	    return (0); | ||||
| 	} | ||||
|       return (FNM_NOMATCH); | ||||
|     } | ||||
| 
 | ||||
|   return (FNM_NOMATCH); | ||||
| } | ||||
| #endif /* EXTENDED_GLOB */ | ||||
| 
 | ||||
| #undef IS_CCLASS | ||||
| #undef FOLD | ||||
| #undef CHAR | ||||
| #undef U_CHAR | ||||
| #undef XCHAR | ||||
| #undef INT | ||||
| #undef INVALID | ||||
| #undef FCT | ||||
| #undef GMATCH | ||||
| #undef COLLSYM | ||||
| #undef PARSE_COLLSYM | ||||
| #undef PATSCAN | ||||
| #undef STRCOMPARE | ||||
| #undef EXTMATCH | ||||
| #undef BRACKMATCH | ||||
| #undef STRCHR | ||||
| #undef STRCOLL | ||||
| #undef STRLEN | ||||
| #undef STRCMP | ||||
| #undef COLLEQUIV | ||||
| #undef RANGECMP | ||||
| #undef L | ||||
							
								
								
									
										410
									
								
								lib/glob/smatch.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										410
									
								
								lib/glob/smatch.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,410 @@ | |||
| /* strmatch.c -- ksh-like extended pattern matching for the shell and filename
 | ||||
| 		globbing. */ | ||||
| 
 | ||||
| /* Copyright (C) 1991-2002 Free Software Foundation, Inc.
 | ||||
| 
 | ||||
|    This file is part of GNU Bash, the Bourne Again SHell. | ||||
|     | ||||
|    Bash 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, or (at your option) any later | ||||
|    version. | ||||
| 	       | ||||
|    Bash 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 Bash; see the file COPYING.  If not, write to the Free Software | ||||
|    Foundation, 59 Temple Place, Suite 330, Boston, MA 02111 USA. */ | ||||
| 
 | ||||
| #include <config.h> | ||||
| 
 | ||||
| #include <stdio.h>	/* for debugging */ | ||||
| 				 | ||||
| #include "strmatch.h" | ||||
| #include <chartypes.h> | ||||
| 
 | ||||
| #include "bashansi.h" | ||||
| #include "shmbutil.h" | ||||
| #include "xmalloc.h" | ||||
| 
 | ||||
| /* First, compile `sm_loop.c' for single-byte characters. */ | ||||
| #define CHAR	unsigned char | ||||
| #define U_CHAR	unsigned char | ||||
| #define XCHAR	char | ||||
| #define INT	int | ||||
| #define L(CS)	CS | ||||
| #define INVALID	-1 | ||||
| 
 | ||||
| #undef STREQ | ||||
| #undef STREQN | ||||
| #define STREQ(a, b) ((a)[0] == (b)[0] && strcmp(a, b) == 0) | ||||
| #define STREQN(a, b, n) ((a)[0] == (b)[0] && strncmp(a, b, n) == 0) | ||||
| 
 | ||||
| /* We use strcoll(3) for range comparisons in bracket expressions,
 | ||||
|    even though it can have unwanted side effects in locales | ||||
|    other than POSIX or US.  For instance, in the de locale, [A-Z] matches | ||||
|    all characters. */ | ||||
| 
 | ||||
| #if defined (HAVE_STRCOLL) | ||||
| /* Helper function for collating symbol equivalence. */ | ||||
| static int rangecmp (c1, c2) | ||||
|      int c1, c2; | ||||
| { | ||||
|   static char s1[2] = { ' ', '\0' }; | ||||
|   static char s2[2] = { ' ', '\0' }; | ||||
|   int ret; | ||||
| 
 | ||||
|   /* Eight bits only.  Period. */ | ||||
|   c1 &= 0xFF; | ||||
|   c2 &= 0xFF; | ||||
| 
 | ||||
|   if (c1 == c2) | ||||
|     return (0); | ||||
| 
 | ||||
|   s1[0] = c1; | ||||
|   s2[0] = c2; | ||||
| 
 | ||||
|   if ((ret = strcoll (s1, s2)) != 0) | ||||
|     return ret; | ||||
|   return (c1 - c2); | ||||
| } | ||||
| #else /* !HAVE_STRCOLL */ | ||||
| #  define rangecmp(c1, c2)	((int)(c1) - (int)(c2)) | ||||
| #endif /* !HAVE_STRCOLL */ | ||||
| 
 | ||||
| #if defined (HAVE_STRCOLL) | ||||
| static int | ||||
| collequiv (c1, c2) | ||||
|      int c1, c2; | ||||
| { | ||||
|   return (rangecmp (c1, c2) == 0); | ||||
| } | ||||
| #else | ||||
| #  define collequiv(c1, c2)	((c1) == (c2)) | ||||
| #endif | ||||
| 
 | ||||
| #define _COLLSYM	_collsym | ||||
| #define __COLLSYM	__collsym | ||||
| #define POSIXCOLL	posix_collsyms | ||||
| #include "collsyms.h" | ||||
| 
 | ||||
| static int | ||||
| collsym (s, len) | ||||
|      char *s; | ||||
|      int len; | ||||
| { | ||||
|   register struct _collsym *csp; | ||||
| 
 | ||||
|   for (csp = posix_collsyms; csp->name; csp++) | ||||
|     { | ||||
|       if (STREQN(csp->name, s, len) && csp->name[len] == '\0') | ||||
| 	return (csp->code); | ||||
|     } | ||||
|   if (len == 1) | ||||
|     return s[0]; | ||||
|   return INVALID; | ||||
| } | ||||
| 
 | ||||
| /* unibyte character classification */ | ||||
| #if !defined (isascii) && !defined (HAVE_ISASCII) | ||||
| #  define isascii(c)	((unsigned int)(c) <= 0177) | ||||
| #endif | ||||
| 
 | ||||
| enum char_class | ||||
|   { | ||||
|     CC_NO_CLASS = 0, | ||||
|     CC_ASCII, CC_ALNUM, CC_ALPHA, CC_BLANK, CC_CNTRL, CC_DIGIT, CC_GRAPH, | ||||
|     CC_LOWER, CC_PRINT, CC_PUNCT, CC_SPACE, CC_UPPER, CC_WORD, CC_XDIGIT | ||||
|   }; | ||||
| 
 | ||||
| static char const *const cclass_name[] = | ||||
|   { | ||||
|     "", | ||||
|     "ascii", "alnum", "alpha", "blank", "cntrl", "digit", "graph", | ||||
|     "lower", "print", "punct", "space", "upper", "word", "xdigit" | ||||
|   }; | ||||
| 
 | ||||
| #define N_CHAR_CLASS (sizeof(cclass_name) / sizeof (cclass_name[0])) | ||||
| 
 | ||||
| static int | ||||
| is_cclass (c, name) | ||||
|      int c; | ||||
|      const char *name; | ||||
| { | ||||
|   enum char_class char_class = CC_NO_CLASS; | ||||
|   int i, result; | ||||
| 
 | ||||
|   for (i = 1; i < N_CHAR_CLASS; i++) | ||||
|     { | ||||
|       if (STREQ (name, cclass_name[i])) | ||||
| 	{ | ||||
| 	  char_class = (enum char_class)i; | ||||
| 	  break; | ||||
| 	} | ||||
|     } | ||||
| 
 | ||||
|   if (char_class == 0) | ||||
|     return -1; | ||||
| 
 | ||||
|   switch (char_class) | ||||
|     { | ||||
|       case CC_ASCII: | ||||
| 	result = isascii (c); | ||||
| 	break; | ||||
|       case CC_ALNUM: | ||||
| 	result = ISALNUM (c); | ||||
| 	break; | ||||
|       case CC_ALPHA: | ||||
| 	result = ISALPHA (c); | ||||
| 	break; | ||||
|       case CC_BLANK:   | ||||
| 	result = ISBLANK (c); | ||||
| 	break; | ||||
|       case CC_CNTRL: | ||||
| 	result = ISCNTRL (c); | ||||
| 	break; | ||||
|       case CC_DIGIT: | ||||
| 	result = ISDIGIT (c); | ||||
| 	break; | ||||
|       case CC_GRAPH: | ||||
| 	result = ISGRAPH (c); | ||||
| 	break; | ||||
|       case CC_LOWER: | ||||
| 	result = ISLOWER (c); | ||||
| 	break; | ||||
|       case CC_PRINT:  | ||||
| 	result = ISPRINT (c); | ||||
| 	break; | ||||
|       case CC_PUNCT: | ||||
| 	result = ISPUNCT (c); | ||||
| 	break; | ||||
|       case CC_SPACE: | ||||
| 	result = ISSPACE (c); | ||||
| 	break; | ||||
|       case CC_UPPER: | ||||
| 	result = ISUPPER (c); | ||||
| 	break; | ||||
|       case CC_WORD: | ||||
|         result = (ISALNUM (c) || c == '_'); | ||||
| 	break; | ||||
|       case CC_XDIGIT: | ||||
| 	result = ISXDIGIT (c); | ||||
| 	break; | ||||
|       default: | ||||
| 	result = -1; | ||||
| 	break; | ||||
|     } | ||||
| 
 | ||||
|   return result;   | ||||
| } | ||||
| 
 | ||||
| /* Now include `sm_loop.c' for single-byte characters. */ | ||||
| /* The result of FOLD is an `unsigned char' */ | ||||
| # define FOLD(c) ((flags & FNM_CASEFOLD) \ | ||||
| 	? TOLOWER ((unsigned char)c) \ | ||||
| 	: ((unsigned char)c)) | ||||
| 
 | ||||
| #define FCT			internal_strmatch | ||||
| #define GMATCH			gmatch | ||||
| #define COLLSYM			collsym | ||||
| #define PARSE_COLLSYM		parse_collsym | ||||
| #define BRACKMATCH		brackmatch | ||||
| #define PATSCAN			patscan | ||||
| #define STRCOMPARE		strcompare | ||||
| #define EXTMATCH		extmatch | ||||
| #define STRCHR(S, C)		strchr((S), (C)) | ||||
| #define STRCOLL(S1, S2)		strcoll((S1), (S2)) | ||||
| #define STRLEN(S)		strlen(S) | ||||
| #define STRCMP(S1, S2)		strcmp((S1), (S2)) | ||||
| #define RANGECMP(C1, C2)	rangecmp((C1), (C2)) | ||||
| #define COLLEQUIV(C1, C2)	collequiv((C1), (C2)) | ||||
| #define CTYPE_T			enum char_class | ||||
| #define IS_CCLASS(C, S)		is_cclass((C), (S)) | ||||
| #include "sm_loop.c" | ||||
| 
 | ||||
| #if HANDLE_MULTIBYTE | ||||
| 
 | ||||
| #  define CHAR		wchar_t | ||||
| #  define U_CHAR	wint_t | ||||
| #  define XCHAR		wchar_t | ||||
| #  define INT		wint_t | ||||
| #  define L(CS)		L##CS | ||||
| #  define INVALID	WEOF | ||||
| 
 | ||||
| #  undef STREQ | ||||
| #  undef STREQN | ||||
| #  define STREQ(s1, s2) ((wcscmp (s1, s2) == 0)) | ||||
| #  define STREQN(a, b, n) ((a)[0] == (b)[0] && wcsncmp(a, b, n) == 0) | ||||
| 
 | ||||
| static int | ||||
| rangecmp_wc (c1, c2) | ||||
|      wint_t c1, c2; | ||||
| { | ||||
|   static wchar_t s1[2] = { L' ', L'\0' }; | ||||
|   static wchar_t s2[2] = { L' ', L'\0' }; | ||||
|   int ret; | ||||
| 
 | ||||
|   if (c1 == c2) | ||||
|     return 0; | ||||
| 
 | ||||
|   s1[0] = c1; | ||||
|   s2[0] = c2; | ||||
| 
 | ||||
|   return (wcscoll (s1, s2)); | ||||
| } | ||||
| 
 | ||||
| static int | ||||
| collequiv_wc (c, equiv) | ||||
|      wint_t c, equiv; | ||||
| { | ||||
|   return (!(c - equiv)); | ||||
| } | ||||
| 
 | ||||
| /* Helper function for collating symbol. */ | ||||
| #  define _COLLSYM	_collwcsym | ||||
| #  define __COLLSYM	__collwcsym | ||||
| #  define POSIXCOLL	posix_collwcsyms | ||||
| #  include "collsyms.h" | ||||
| 
 | ||||
| static wint_t | ||||
| collwcsym (s, len) | ||||
|      wchar_t *s; | ||||
|      int len; | ||||
| { | ||||
|   register struct _collwcsym *csp; | ||||
| 
 | ||||
|   for (csp = posix_collwcsyms; csp->name; csp++) | ||||
|     { | ||||
|       if (STREQN(csp->name, s, len) && csp->name[len] == L'\0') | ||||
| 	return (csp->code); | ||||
|     } | ||||
|   if (len == 1) | ||||
|     return s[0]; | ||||
|   return INVALID; | ||||
| } | ||||
| 
 | ||||
| static int | ||||
| is_wcclass (wc, name) | ||||
|      wint_t wc; | ||||
|      wchar_t *name; | ||||
| { | ||||
|   char *mbs; | ||||
|   mbstate_t state; | ||||
|   size_t mbslength; | ||||
|   wctype_t desc; | ||||
|   int want_word; | ||||
| 
 | ||||
|   if ((wctype ("ascii") == (wctype_t)0) && (wcscmp (name, L"ascii") == 0)) | ||||
|     { | ||||
|       int c; | ||||
| 
 | ||||
|       if ((c = wctob (wc)) == EOF) | ||||
| 	return 0; | ||||
|       else | ||||
|         return (c <= 0x7F); | ||||
|     } | ||||
| 
 | ||||
|   want_word = (wcscmp (name, L"word") == 0); | ||||
|   if (want_word) | ||||
|     name = L"alnum"; | ||||
| 
 | ||||
|   memset (&state, '\0', sizeof (mbstate_t)); | ||||
|   mbs = (char *) malloc (wcslen(name) * MB_CUR_MAX + 1); | ||||
|   mbslength = wcsrtombs(mbs, (const wchar_t **)&name, (wcslen(name) * MB_CUR_MAX + 1), &state); | ||||
| 
 | ||||
|   if (mbslength == (size_t)-1 || mbslength == (size_t)-2) | ||||
|     { | ||||
|       free (mbs); | ||||
|       return -1; | ||||
|     } | ||||
|   desc = wctype (mbs); | ||||
|   free (mbs); | ||||
| 
 | ||||
|   if (desc == (wctype_t)0) | ||||
|     return -1; | ||||
| 
 | ||||
|   if (want_word) | ||||
|     return (iswctype (wc, desc) || wc == L'_'); | ||||
|   else | ||||
|     return (iswctype (wc, desc)); | ||||
| } | ||||
| 
 | ||||
| /* Now include `sm_loop.c' for multibyte characters. */ | ||||
| #define FOLD(c) ((flags & FNM_CASEFOLD) && iswupper (c) ? towlower (c) : (c)) | ||||
| #define FCT			internal_wstrmatch | ||||
| #define GMATCH			gmatch_wc | ||||
| #define COLLSYM			collwcsym | ||||
| #define PARSE_COLLSYM		parse_collwcsym | ||||
| #define BRACKMATCH		brackmatch_wc | ||||
| #define PATSCAN			patscan_wc | ||||
| #define STRCOMPARE		wscompare | ||||
| #define EXTMATCH		extmatch_wc | ||||
| #define STRCHR(S, C)		wcschr((S), (C)) | ||||
| #define STRCOLL(S1, S2)		wcscoll((S1), (S2)) | ||||
| #define STRLEN(S)		wcslen(S) | ||||
| #define STRCMP(S1, S2)		wcscmp((S1), (S2)) | ||||
| #define RANGECMP(C1, C2)	rangecmp_wc((C1), (C2)) | ||||
| #define COLLEQUIV(C1, C2)	collequiv_wc((C1), (C2)) | ||||
| #define CTYPE_T			enum char_class | ||||
| #define IS_CCLASS(C, S)		is_wcclass((C), (S)) | ||||
| #include "sm_loop.c" | ||||
| 
 | ||||
| #endif /* HAVE_MULTIBYTE */ | ||||
| 
 | ||||
| int | ||||
| xstrmatch (pattern, string, flags) | ||||
|      char *pattern; | ||||
|      char *string; | ||||
|      int flags; | ||||
| { | ||||
| #if HANDLE_MULTIBYTE | ||||
|   int ret; | ||||
|   mbstate_t ps; | ||||
|   size_t n; | ||||
|   char *pattern_bak; | ||||
|   wchar_t *wpattern, *wstring; | ||||
| 
 | ||||
|   if (MB_CUR_MAX == 1) | ||||
|     return (internal_strmatch (pattern, string, flags)); | ||||
| 
 | ||||
|   pattern_bak = (char *)xmalloc (strlen (pattern) + 1); | ||||
|   strcpy (pattern_bak, pattern); | ||||
| 
 | ||||
|   memset (&ps, '\0', sizeof (mbstate_t)); | ||||
|   n = xmbsrtowcs (NULL, (const char **)&pattern, 0, &ps); | ||||
|   if (n == (size_t)-1 || n == (size_t)-2) | ||||
|     { | ||||
|       free (pattern_bak); | ||||
|       return (internal_strmatch ((unsigned char *)pattern, (unsigned char *)string, flags)); | ||||
|     } | ||||
| 
 | ||||
|   wpattern = (wchar_t *)xmalloc ((n + 1) * sizeof (wchar_t)); | ||||
|   (void) xmbsrtowcs (wpattern, (const char **)&pattern, n + 1, &ps); | ||||
| 
 | ||||
|   memset (&ps, '\0', sizeof (mbstate_t)); | ||||
|   n = xmbsrtowcs (NULL, (const char **)&string, 0, &ps); | ||||
|   if (n == (size_t)-1 || n == (size_t)-2) | ||||
|     { | ||||
|       free (wpattern); | ||||
|       ret = internal_strmatch (pattern_bak, string, flags); | ||||
|       free (pattern_bak); | ||||
|       return ret; | ||||
|     } | ||||
| 
 | ||||
|   wstring = (wchar_t *)xmalloc ((n + 1) * sizeof (wchar_t)); | ||||
|   (void) xmbsrtowcs (wstring, (const char **)&string, n + 1, &ps); | ||||
| 
 | ||||
|   ret = internal_wstrmatch (wpattern, wstring, flags); | ||||
| 
 | ||||
|   free (pattern_bak); | ||||
|   free (wpattern); | ||||
|   free (wstring); | ||||
| 
 | ||||
|   return ret; | ||||
| #else | ||||
|   return (internal_strmatch ((unsigned char *)pattern, (unsigned char *)string, flags)); | ||||
| #endif /* !HANDLE_MULTIBYTE */ | ||||
| } | ||||
|  | @ -1,7 +1,7 @@ | |||
| /* strmatch.c -- ksh-like extended pattern matching for the shell and filename
 | ||||
| 		globbing. */ | ||||
| 
 | ||||
| /* Copyright (C) 1991, 1997 Free Software Foundation, Inc.
 | ||||
| /* Copyright (C) 1991-2002 Free Software Foundation, Inc.
 | ||||
| 
 | ||||
|    This file is part of GNU Bash, the Bourne Again SHell. | ||||
|     | ||||
|  | @ -21,805 +21,29 @@ | |||
| 
 | ||||
| #include <config.h> | ||||
| 
 | ||||
| #include <stdio.h>	/* for debugging */ | ||||
| 				 | ||||
| #include "stdc.h" | ||||
| #include "strmatch.h" | ||||
| #include "collsyms.h" | ||||
| #include <chartypes.h> | ||||
| 
 | ||||
| #if defined (HAVE_STRING_H) | ||||
| #  include <string.h> | ||||
| /* Structured this way so that if HAVE_LIBC_FNM_EXTMATCH is defined, the
 | ||||
|    matching portion of the library (smatch.c) is not linked into the shell. */ | ||||
| 
 | ||||
| #ifndef HAVE_LIBC_FNM_EXTMATCH | ||||
| extern int xstrmatch __P((char *, char *, int)); | ||||
| #else | ||||
| #  include <strings.h> | ||||
| #endif /* HAVE_STRING_H */ | ||||
| 
 | ||||
| static int gmatch (); | ||||
| static char *brackmatch (); | ||||
| #ifdef EXTENDED_GLOB | ||||
| static int extmatch (); | ||||
| static char *patscan (); | ||||
| #endif | ||||
|    | ||||
| #if !defined (isascii) && !defined (HAVE_ISASCII) | ||||
| #  define isascii(c)	((unsigned int)(c) <= 0177) | ||||
| #  define xstrmatch fnmatch | ||||
| #endif | ||||
| 
 | ||||
| /* The result of FOLD is an `unsigned char' */ | ||||
| # define FOLD(c) ((flags & FNM_CASEFOLD) \ | ||||
| 	? TOLOWER ((unsigned char)c) \ | ||||
| 	: ((unsigned char)c)) | ||||
| 
 | ||||
| #ifndef STREQ | ||||
| #define STREQ(a, b) ((a)[0] == (b)[0] && strcmp(a, b) == 0) | ||||
| #define STREQN(a, b, n) ((a)[0] == (b)[0] && strncmp(a, b, n) == 0) | ||||
| #endif | ||||
| 
 | ||||
| /* We use strcoll(3) for range comparisons in bracket expressions,
 | ||||
|    even though it can have unwanted side effects in locales | ||||
|    other than POSIX or US.  For instance, in the de locale, [A-Z] matches | ||||
|    all characters. */ | ||||
| 
 | ||||
| #if defined (HAVE_STRCOLL) | ||||
| /* Helper function for collating symbol equivalence. */ | ||||
| static int rangecmp (c1, c2) | ||||
|      int c1, c2; | ||||
| { | ||||
|   static char s1[2] = { ' ', '\0' }; | ||||
|   static char s2[2] = { ' ', '\0' }; | ||||
|   int ret; | ||||
| 
 | ||||
|   /* Eight bits only.  Period. */ | ||||
|   c1 &= 0xFF; | ||||
|   c2 &= 0xFF; | ||||
| 
 | ||||
|   if (c1 == c2) | ||||
|     return (0); | ||||
| 
 | ||||
|   s1[0] = c1; | ||||
|   s2[0] = c2; | ||||
| 
 | ||||
|   if ((ret = strcoll (s1, s2)) != 0) | ||||
|     return ret; | ||||
|   return (c1 - c2); | ||||
| } | ||||
| #else /* !HAVE_STRCOLL */ | ||||
| #  define rangecmp(c1, c2)	((int)(c1) - (int)(c2)) | ||||
| #endif /* !HAVE_STRCOLL */ | ||||
| 
 | ||||
| #if defined (HAVE_STRCOLL) | ||||
| static int collequiv (c1, c2) | ||||
|      int c1, c2; | ||||
| { | ||||
|   return (rangecmp (c1, c2) == 0); | ||||
| } | ||||
| #else | ||||
| #  define collequiv(c1, c2)	((c1) == (c2)) | ||||
| #endif | ||||
| 
 | ||||
| static int | ||||
| collsym (s, len) | ||||
|      char *s; | ||||
|      int len; | ||||
| { | ||||
|   register struct _collsym *csp; | ||||
| 
 | ||||
|   for (csp = posix_collsyms; csp->name; csp++) | ||||
|     { | ||||
|       if (STREQN(csp->name, s, len) && csp->name[len] == '\0') | ||||
| 	return (csp->code); | ||||
|     } | ||||
|   if (len == 1) | ||||
|     return s[0]; | ||||
|   return -1; | ||||
| } | ||||
| 
 | ||||
| #ifdef HAVE_LIBC_FNM_EXTMATCH | ||||
| int | ||||
| strmatch (pattern, string, flags) | ||||
|      char *pattern; | ||||
|      char *string; | ||||
|      int flags; | ||||
| { | ||||
|   char *se, *pe; | ||||
| 
 | ||||
|   if (string == 0 || pattern == 0) | ||||
|     return FNM_NOMATCH; | ||||
| 
 | ||||
|   return (fnmatch (pattern, string, flags)); | ||||
|   return (xstrmatch (pattern, string, flags)); | ||||
| } | ||||
| #else /* !HAVE_LIBC_FNM_EXTMATCH */ | ||||
| int | ||||
| strmatch (pattern, string, flags) | ||||
|      char *pattern; | ||||
|      char *string; | ||||
|      int flags; | ||||
| { | ||||
|   char *se, *pe; | ||||
| 
 | ||||
|   if (string == 0 || pattern == 0) | ||||
|     return FNM_NOMATCH; | ||||
| 
 | ||||
|   se = string + strlen (string); | ||||
|   pe = pattern + strlen (pattern); | ||||
| 
 | ||||
|   return (gmatch (string, se, pattern, pe, flags)); | ||||
| } | ||||
| #endif /* !HAVE_LIBC_FNM_EXTMATCH */ | ||||
| 
 | ||||
| /* Match STRING against the filename pattern PATTERN, returning zero if
 | ||||
|    it matches, FNM_NOMATCH if not.  */ | ||||
| static int | ||||
| gmatch (string, se, pattern, pe, flags) | ||||
|      char *string, *se; | ||||
|      char *pattern, *pe; | ||||
|      int flags; | ||||
| { | ||||
|   register char *p, *n;		/* pattern, string */ | ||||
|   register char c;		/* current pattern character */ | ||||
|   register char sc;		/* current string character */ | ||||
| 
 | ||||
|   p = pattern; | ||||
|   n = string; | ||||
| 
 | ||||
|   if (string == 0 || pattern == 0) | ||||
|     return FNM_NOMATCH; | ||||
| 
 | ||||
| #if DEBUG_MATCHING | ||||
| fprintf(stderr, "gmatch: string = %s; se = %s\n", string, se); | ||||
| fprintf(stderr, "gmatch: pattern = %s; pe = %s\n", pattern, pe); | ||||
| #endif | ||||
| 
 | ||||
|   while (p < pe) | ||||
|     { | ||||
|       c = *p++; | ||||
|       c = FOLD (c); | ||||
| 
 | ||||
|       sc = n < se ? *n : '\0'; | ||||
| 
 | ||||
| #ifdef EXTENDED_GLOB | ||||
|       /* extmatch () will handle recursively calling gmatch, so we can
 | ||||
| 	 just return what extmatch() returns. */ | ||||
|       if ((flags & FNM_EXTMATCH) && *p == '(' && | ||||
| 	  (c == '+' || c == '*' || c == '?' || c == '@' || c == '!')) /* ) */ | ||||
| 	{ | ||||
| 	  int lflags; | ||||
| 	  /* If we're not matching the start of the string, we're not
 | ||||
| 	     concerned about the special cases for matching `.' */ | ||||
| 	  lflags = (n == string) ? flags : (flags & ~FNM_PERIOD); | ||||
| 	  return (extmatch (c, n, se, p, pe, lflags)); | ||||
| 	} | ||||
| #endif | ||||
| 
 | ||||
|       switch (c) | ||||
| 	{ | ||||
| 	case '?':		/* Match single character */ | ||||
| 	  if (sc == '\0') | ||||
| 	    return FNM_NOMATCH; | ||||
| 	  else if ((flags & FNM_PATHNAME) && sc == '/') | ||||
| 	    /* If we are matching a pathname, `?' can never match a `/'. */ | ||||
| 	    return FNM_NOMATCH; | ||||
| 	  else if ((flags & FNM_PERIOD) && sc == '.' && | ||||
| 		   (n == string || ((flags & FNM_PATHNAME) && n[-1] == '/'))) | ||||
| 	    /* `?' cannot match a `.' if it is the first character of the
 | ||||
| 	       string or if it is the first character following a slash and | ||||
| 	       we are matching a pathname. */ | ||||
| 	    return FNM_NOMATCH; | ||||
| 	  break; | ||||
| 
 | ||||
| 	case '\\':		/* backslash escape removes special meaning */ | ||||
| 	  if (p == pe) | ||||
| 	    return FNM_NOMATCH; | ||||
| 
 | ||||
| 	  if ((flags & FNM_NOESCAPE) == 0) | ||||
| 	    { | ||||
| 	      c = *p++; | ||||
| 	      /* A trailing `\' cannot match. */ | ||||
| 	      if (p > pe) | ||||
| 		return FNM_NOMATCH; | ||||
| 	      c = FOLD (c); | ||||
| 	    } | ||||
| 	  if (FOLD (sc) != (unsigned char)c) | ||||
| 	    return FNM_NOMATCH; | ||||
| 	  break; | ||||
| 
 | ||||
| 	case '*':		/* Match zero or more characters */ | ||||
| 	  if (p == pe) | ||||
| 	    return 0; | ||||
| 	   | ||||
| 	  if ((flags & FNM_PERIOD) && sc == '.' && | ||||
| 	      (n == string || ((flags & FNM_PATHNAME) && n[-1] == '/'))) | ||||
| 	    /* `*' cannot match a `.' if it is the first character of the
 | ||||
| 	       string or if it is the first character following a slash and | ||||
| 	       we are matching a pathname. */ | ||||
| 	    return FNM_NOMATCH; | ||||
| 
 | ||||
| 	  /* Collapse multiple consecutive `*' and `?', but make sure that
 | ||||
| 	     one character of the string is consumed for each `?'. */ | ||||
| 	  for (c = *p++; (c == '?' || c == '*'); c = *p++) | ||||
| 	    { | ||||
| 	      if ((flags & FNM_PATHNAME) && sc == '/') | ||||
| 		/* A slash does not match a wildcard under FNM_PATHNAME. */ | ||||
| 		return FNM_NOMATCH; | ||||
| 	      else if (c == '?') | ||||
| 		{ | ||||
| 		  if (sc == '\0') | ||||
| 		    return FNM_NOMATCH; | ||||
| 		  /* One character of the string is consumed in matching
 | ||||
| 		     this ? wildcard, so *??? won't match if there are | ||||
| 		     fewer than three characters. */ | ||||
| 		  n++; | ||||
| 		  sc = n < se ? *n : '\0'; | ||||
| 		} | ||||
| 
 | ||||
| #ifdef EXTENDED_GLOB | ||||
| 	      /* Handle ******(patlist) */ | ||||
| 	      if ((flags & FNM_EXTMATCH) && c == '*' && *p == '(')  /*)*/ | ||||
| 		{ | ||||
| 		  char *newn; | ||||
| 		  /* We need to check whether or not the extended glob
 | ||||
| 		     pattern matches the remainder of the string. | ||||
| 		     If it does, we match the entire pattern. */ | ||||
| 		  for (newn = n; newn < se; ++newn) | ||||
| 		    { | ||||
| 		      if (extmatch (c, newn, se, p, pe, flags) == 0) | ||||
| 			return (0); | ||||
| 		    } | ||||
| 		  /* We didn't match the extended glob pattern, but
 | ||||
| 		     that's OK, since we can match 0 or more occurrences. | ||||
| 		     We need to skip the glob pattern and see if we | ||||
| 		     match the rest of the string. */ | ||||
| 		  newn = patscan (p + 1, pe, 0); | ||||
| 		  /* If NEWN is 0, we have an ill-formed pattern. */ | ||||
| 		  p = newn ? newn : pe; | ||||
| 		} | ||||
| #endif | ||||
| 	      if (p == pe) | ||||
| 		break; | ||||
| 	    } | ||||
| 
 | ||||
| 	  /* If we've hit the end of the pattern and the last character of
 | ||||
| 	     the pattern was handled by the loop above, we've succeeded. | ||||
| 	     Otherwise, we need to match that last character. */ | ||||
| 	  if (p == pe && (c == '?' || c == '*')) | ||||
| 	    return (0); | ||||
| 
 | ||||
| 	  /* General case, use recursion. */ | ||||
| 	  { | ||||
| 	    unsigned char c1; | ||||
| 
 | ||||
| 	    c1 = (unsigned char)((flags & FNM_NOESCAPE) == 0 && c == '\\') ? *p : c; | ||||
| 	    c1 = FOLD (c1); | ||||
| 	    for (--p; n < se; ++n) | ||||
| 	      { | ||||
| 		/* Only call strmatch if the first character indicates a
 | ||||
| 		   possible match.  We can check the first character if | ||||
| 		   we're not doing an extended glob match. */ | ||||
| 		if ((flags & FNM_EXTMATCH) == 0 && c != '[' && FOLD (*n) != c1) /*]*/ | ||||
| 		  continue; | ||||
| 
 | ||||
| 		/* If we're doing an extended glob match and the pattern is not
 | ||||
| 		   one of the extended glob patterns, we can check the first | ||||
| 		   character. */ | ||||
| 		if ((flags & FNM_EXTMATCH) && p[1] != '(' && /*)*/ | ||||
| 		    strchr ("?*+@!", *p) == 0 && c != '[' && FOLD (*n) != c1) /*]*/ | ||||
| 		  continue; | ||||
| 
 | ||||
| 		/* Otherwise, we just recurse. */ | ||||
| 		if (gmatch (n, se, p, pe, flags & ~FNM_PERIOD) == 0) | ||||
| 		  return (0); | ||||
| 	      } | ||||
| 	    return FNM_NOMATCH; | ||||
| 	  } | ||||
| 
 | ||||
| 	case '[': | ||||
| 	  { | ||||
| 	    if (sc == '\0' || n == se) | ||||
| 	      return FNM_NOMATCH; | ||||
| 
 | ||||
| 	    /* A character class cannot match a `.' if it is the first
 | ||||
| 	       character of the string or if it is the first character | ||||
| 	       following a slash and we are matching a pathname. */ | ||||
| 	    if ((flags & FNM_PERIOD) && sc == '.' && | ||||
| 		(n == string || ((flags & FNM_PATHNAME) && n[-1] == '/'))) | ||||
| 	      return (FNM_NOMATCH); | ||||
| 
 | ||||
| 	    p = brackmatch (p, sc, flags); | ||||
| 	    if (p == 0) | ||||
| 	      return FNM_NOMATCH; | ||||
| 	  } | ||||
| 	  break; | ||||
| 
 | ||||
| 	default: | ||||
| 	  if ((unsigned char)c != FOLD (sc)) | ||||
| 	    return (FNM_NOMATCH); | ||||
| 	} | ||||
| 
 | ||||
|       ++n; | ||||
|     } | ||||
| 
 | ||||
|   if (n == se) | ||||
|     return (0); | ||||
| 
 | ||||
|   if ((flags & FNM_LEADING_DIR) && *n == '/') | ||||
|     /* The FNM_LEADING_DIR flag says that "foo*" matches "foobar/frobozz".  */ | ||||
|     return 0; | ||||
| 	   | ||||
|   return (FNM_NOMATCH); | ||||
| } | ||||
| 
 | ||||
| /* Parse a bracket expression collating symbol ([.sym.]) starting at P, find
 | ||||
|    the value of the symbol, and move P past the collating symbol expression. | ||||
|    The value is returned in *VP, if VP is not null. */ | ||||
| static char * | ||||
| parse_collsym (p, vp) | ||||
|      char *p; | ||||
|      int *vp; | ||||
| { | ||||
|   register int pc; | ||||
|   int val; | ||||
| 
 | ||||
|   p++;				/* move past the `.' */ | ||||
| 	   | ||||
|   for (pc = 0; p[pc]; pc++) | ||||
|     if (p[pc] == '.' && p[pc+1] == ']') | ||||
|       break; | ||||
|    val = collsym (p, pc); | ||||
|    if (vp) | ||||
|      *vp = val; | ||||
|    return (p + pc + 2); | ||||
| } | ||||
| 
 | ||||
| static char * | ||||
| brackmatch (p, test, flags) | ||||
|      char *p; | ||||
|      unsigned char test; | ||||
|      int flags; | ||||
| { | ||||
|   register char cstart, cend, c; | ||||
|   register int not;    /* Nonzero if the sense of the character class is inverted.  */ | ||||
|   int pc, brcnt; | ||||
|   char *savep; | ||||
| 
 | ||||
|   test = FOLD (test); | ||||
| 
 | ||||
|   savep = p; | ||||
| 
 | ||||
|   /* POSIX.2 3.13.1 says that an exclamation mark (`!') shall replace the
 | ||||
|      circumflex (`^') in its role in a `nonmatching list'.  A bracket | ||||
|      expression starting with an unquoted circumflex character produces | ||||
|      unspecified results.  This implementation treats the two identically. */ | ||||
|   if (not = (*p == '!' || *p == '^')) | ||||
|     ++p; | ||||
| 
 | ||||
|   c = *p++; | ||||
|   for (;;) | ||||
|     { | ||||
|       /* Initialize cstart and cend in case `-' is the last
 | ||||
| 	 character of the pattern. */ | ||||
|       cstart = cend = c; | ||||
| 
 | ||||
|       /* POSIX.2 equivalence class:  [=c=].  See POSIX.2 2.8.3.2.  Find
 | ||||
| 	 the end of the equivalence class, move the pattern pointer past | ||||
| 	 it, and check for equivalence.  XXX - this handles only | ||||
| 	 single-character equivalence classes, which is wrong, or at | ||||
| 	 least incomplete. */ | ||||
|       if (c == '[' && *p == '=' && p[2] == '=' && p[3] == ']') | ||||
| 	{ | ||||
| 	  pc = FOLD (p[1]); | ||||
| 	  p += 4; | ||||
| 	  if (collequiv (test, pc)) | ||||
| 	    { | ||||
| /*[*/	      /* Move past the closing `]', since the first thing we do at
 | ||||
| 		 the `matched:' label is back p up one. */ | ||||
| 	      p++; | ||||
| 	      goto matched; | ||||
| 	    } | ||||
| 	  else | ||||
| 	    { | ||||
| 	      c = *p++; | ||||
| 	      if (c == '\0') | ||||
| 		return ((test == '[') ? savep : (char *)0); /*]*/ | ||||
| 	      c = FOLD (c); | ||||
| 	      continue; | ||||
| 	    } | ||||
| 	} | ||||
| 
 | ||||
|       /* POSIX.2 character class expression.  See POSIX.2 2.8.3.2. */ | ||||
|       if (c == '[' && *p == ':')	/*]*/ | ||||
| 	{ | ||||
| 	  pc = 0;	/* make sure invalid char classes don't match. */ | ||||
| 	  if (STREQN (p+1, "alnum:]", 7)) | ||||
| 	    { pc = ISALNUM (test); p += 8; } | ||||
| 	  else if (STREQN (p+1, "alpha:]", 7)) | ||||
| 	    { pc = ISALPHA (test); p += 8; } | ||||
| 	  else if (STREQN (p+1, "blank:]", 7)) | ||||
| 	    { pc = ISBLANK (test); p += 8; } | ||||
| 	  else if (STREQN (p+1, "cntrl:]", 7)) | ||||
| 	    { pc = ISCNTRL (test); p += 8; } | ||||
| 	  else if (STREQN (p+1, "digit:]", 7)) | ||||
| 	    { pc = ISDIGIT (test); p += 8; } | ||||
| 	  else if (STREQN (p+1, "graph:]", 7)) | ||||
| 	    { pc = ISGRAPH (test); p += 8; } | ||||
| 	  else if (STREQN (p+1, "lower:]", 7)) | ||||
| 	    { pc = ISLOWER (test); p += 8; } | ||||
| 	  else if (STREQN (p+1, "print:]", 7)) | ||||
| 	    { pc = ISPRINT (test); p += 8; } | ||||
| 	  else if (STREQN (p+1, "punct:]", 7)) | ||||
| 	    { pc = ISPUNCT (test); p += 8; } | ||||
| 	  else if (STREQN (p+1, "space:]", 7)) | ||||
| 	    { pc = ISSPACE (test); p += 8; } | ||||
| 	  else if (STREQN (p+1, "upper:]", 7)) | ||||
| 	    { pc = ISUPPER (test); p += 8; } | ||||
| 	  else if (STREQN (p+1, "xdigit:]", 8)) | ||||
| 	    { pc = ISXDIGIT (test); p += 9; } | ||||
| 	  else if (STREQN (p+1, "ascii:]", 7)) | ||||
| 	    { pc = isascii (test); p += 8; } | ||||
| 	  if (pc) | ||||
| 	    { | ||||
| /*[*/	      /* Move past the closing `]', since the first thing we do at
 | ||||
| 		 the `matched:' label is back p up one. */ | ||||
| 	      p++; | ||||
| 	      goto matched; | ||||
| 	    } | ||||
| 	  else | ||||
| 	    { | ||||
| 	      /* continue the loop here, since this expression can't be
 | ||||
| 		 the first part of a range expression. */ | ||||
| 	      c = *p++; | ||||
| 	      if (c == '\0') | ||||
| 		return ((test == '[') ? savep : (char *)0); | ||||
| 	      else if (c == ']') | ||||
| 		break; | ||||
| 	      c = FOLD (c); | ||||
| 	      continue; | ||||
| 	    } | ||||
| 	} | ||||
|   | ||||
|       /* POSIX.2 collating symbols.  See POSIX.2 2.8.3.2.  Find the end of
 | ||||
| 	 the symbol name, make sure it is terminated by `.]', translate | ||||
| 	 the name to a character using the external table, and do the | ||||
| 	 comparison. */ | ||||
|       if (c == '[' && *p == '.') | ||||
| 	{ | ||||
| 	  p = parse_collsym (p, &pc); | ||||
| 	  /* An invalid collating symbol cannot be the first point of a
 | ||||
| 	     range.  If it is, we set cstart to one greater than `test', | ||||
| 	     so any comparisons later will fail. */ | ||||
| 	  cstart = (pc == -1) ? test + 1 : pc; | ||||
| 	} | ||||
| 
 | ||||
|       if (!(flags & FNM_NOESCAPE) && c == '\\') | ||||
| 	{ | ||||
| 	  if (*p == '\0') | ||||
| 	    return (char *)0; | ||||
| 	  cstart = cend = *p++; | ||||
| 	} | ||||
| 
 | ||||
|       cstart = cend = FOLD (cstart); | ||||
| 
 | ||||
|       /* POSIX.2 2.8.3.1.2 says: `An expression containing a `[' that
 | ||||
| 	 is not preceded by a backslash and is not part of a bracket | ||||
| 	 expression produces undefined results.'  This implementation | ||||
| 	 treats the `[' as just a character to be matched if there is | ||||
| 	 not a closing `]'. */ | ||||
|       if (c == '\0') | ||||
| 	return ((test == '[') ? savep : (char *)0); | ||||
| 
 | ||||
|       c = *p++; | ||||
|       c = FOLD (c); | ||||
| 
 | ||||
|       if ((flags & FNM_PATHNAME) && c == '/') | ||||
| 	/* [/] can never match when matching a pathname.  */ | ||||
| 	return (char *)0; | ||||
| 
 | ||||
|       /* This introduces a range, unless the `-' is the last
 | ||||
| 	 character of the class.  Find the end of the range | ||||
| 	 and move past it. */ | ||||
|       if (c == '-' && *p != ']') | ||||
| 	{ | ||||
| 	  cend = *p++; | ||||
| 	  if (!(flags & FNM_NOESCAPE) && cend == '\\') | ||||
| 	    cend = *p++; | ||||
| 	  if (cend == '\0') | ||||
| 	    return (char *)0; | ||||
| 	  if (cend == '[' && *p == '.') | ||||
| 	    { | ||||
| 	      p = parse_collsym (p, &pc); | ||||
| 	      /* An invalid collating symbol cannot be the second part of a
 | ||||
| 		 range expression.  If we get one, we set cend to one fewer | ||||
| 		 than the test character to make sure the range test fails. */ | ||||
| 	      cend = (pc == -1) ? test - 1 : pc; | ||||
| 	    } | ||||
| 	  cend = FOLD (cend); | ||||
| 
 | ||||
| 	  c = *p++; | ||||
| 
 | ||||
| 	  /* POSIX.2 2.8.3.2:  ``The ending range point shall collate
 | ||||
| 	     equal to or higher than the starting range point; otherwise | ||||
| 	     the expression shall be treated as invalid.''  Note that this | ||||
| 	     applies to only the range expression; the rest of the bracket | ||||
| 	     expression is still checked for matches. */ | ||||
| 	  if (rangecmp (cstart, cend) > 0) | ||||
| 	    { | ||||
| 	      if (c == ']') | ||||
| 		break; | ||||
| 	      c = FOLD (c); | ||||
| 	      continue; | ||||
| 	    } | ||||
| 	} | ||||
| 
 | ||||
|       if (rangecmp (test, cstart) >= 0 && rangecmp (test, cend) <= 0) | ||||
| 	goto matched; | ||||
| 
 | ||||
|       if (c == ']') | ||||
| 	break; | ||||
|     } | ||||
|   /* No match. */ | ||||
|   return (!not ? (char *)0 : p); | ||||
| 
 | ||||
| matched: | ||||
|   /* Skip the rest of the [...] that already matched.  */ | ||||
| #if 0 | ||||
|   brcnt = (c != ']') + (c == '[' && (*p == '=' || *p == ':' || *p == '.')); | ||||
| #else | ||||
|   c = *--p; | ||||
|   brcnt = 1; | ||||
| #endif | ||||
|   while (brcnt > 0) | ||||
|     { | ||||
|       /* A `[' without a matching `]' is just another character to match. */ | ||||
|       if (c == '\0') | ||||
| 	return ((test == '[') ? savep : (char *)0); | ||||
| 
 | ||||
|       c = *p++; | ||||
|       if (c == '[' && (*p == '=' || *p == ':' || *p == '.')) | ||||
| 	brcnt++; | ||||
|       else if (c == ']') | ||||
| 	brcnt--; | ||||
|       else if (!(flags & FNM_NOESCAPE) && c == '\\') | ||||
| 	{ | ||||
| 	  if (*p == '\0') | ||||
| 	    return (char *)0; | ||||
| 	  /* XXX 1003.2d11 is unclear if this is right. */ | ||||
| 	  ++p; | ||||
| 	} | ||||
|     } | ||||
|   return (not ? (char *)0 : p); | ||||
| } | ||||
| 
 | ||||
| #if defined (EXTENDED_GLOB) | ||||
| /* ksh-like extended pattern matching:
 | ||||
| 
 | ||||
| 	[?*+@!](pat-list) | ||||
| 
 | ||||
|    where pat-list is a list of one or patterns separated by `|'.  Operation | ||||
|    is as follows: | ||||
| 
 | ||||
| 	?(patlist)	match zero or one of the given patterns | ||||
| 	*(patlist)	match zero or more of the given patterns | ||||
| 	+(patlist)	match one or more of the given patterns | ||||
| 	@(patlist)	match exactly one of the given patterns | ||||
| 	!(patlist)	match anything except one of the given patterns | ||||
| */ | ||||
| 
 | ||||
| /* Scan a pattern starting at STRING and ending at END, keeping track of
 | ||||
|    embedded () and [].  If DELIM is 0, we scan until a matching `)' | ||||
|    because we're scanning a `patlist'.  Otherwise, we scan until we see | ||||
|    DELIM.  In all cases, we never scan past END.  The return value is the | ||||
|    first character after the matching DELIM. */ | ||||
| static char * | ||||
| patscan (string, end, delim) | ||||
|      char *string, *end; | ||||
|      int delim; | ||||
| { | ||||
|   int pnest, bnest, cchar; | ||||
|   char *s, c, *bfirst; | ||||
| 
 | ||||
|   pnest = bnest = cchar = 0; | ||||
|   bfirst = 0; | ||||
|   for (s = string; c = *s; s++) | ||||
|     { | ||||
|       if (s >= end) | ||||
| 	return (s); | ||||
|       switch (c) | ||||
| 	{ | ||||
| 	case '\0': | ||||
| 	  return ((char *)0); | ||||
| 
 | ||||
| 	/* `[' is not special inside a bracket expression, but it may
 | ||||
| 	   introduce one of the special POSIX bracket expressions | ||||
| 	   ([.SYM.], [=c=], [: ... :]) that needs special handling. */ | ||||
| 	case '[': | ||||
| 	  if (bnest == 0) | ||||
| 	    { | ||||
| 	      bfirst = s + 1; | ||||
| 	      if (*bfirst == '!' || *bfirst == '^') | ||||
| 		bfirst++; | ||||
| 	      bnest++; | ||||
| 	    } | ||||
| 	  else if (s[1] == ':' || s[1] == '.' || s[1] == '=') | ||||
| 	    cchar = s[1]; | ||||
| 	  break; | ||||
| 
 | ||||
| 	/* `]' is not special if it's the first char (after a leading `!'
 | ||||
| 	   or `^') in a bracket expression or if it's part of one of the | ||||
| 	   special POSIX bracket expressions ([.SYM.], [=c=], [: ... :]) */ | ||||
| 	case ']': | ||||
| 	  if (bnest) | ||||
| 	    { | ||||
| 	      if (cchar && s[-1] == cchar) | ||||
| 		cchar = 0; | ||||
| 	      else if (s != bfirst) | ||||
| 		{ | ||||
| 		  bnest--; | ||||
| 		  bfirst = 0; | ||||
| 		} | ||||
| 	    } | ||||
| 	  break; | ||||
| 
 | ||||
| 	case '(': | ||||
| 	  if (bnest == 0) | ||||
| 	    pnest++; | ||||
| 	  break; | ||||
| 
 | ||||
| 	case ')': | ||||
| #if 0 | ||||
| 	  if (bnest == 0) | ||||
| 	    pnest--; | ||||
| 	  if (pnest <= 0) | ||||
| 	    return ++s; | ||||
| #else | ||||
| 	  if (bnest == 0 && pnest-- <= 0) | ||||
| 	    return ++s; | ||||
| #endif | ||||
| 	  break; | ||||
| 
 | ||||
| 	case '|': | ||||
| 	  if (bnest == 0 && pnest == 0 && delim == '|') | ||||
| 	    return ++s; | ||||
| 	  break; | ||||
| 	} | ||||
|     } | ||||
| 
 | ||||
|   return (char *)0; | ||||
| } | ||||
| 
 | ||||
| /* Return 0 if dequoted pattern matches S in the current locale. */ | ||||
| static int | ||||
| strcompare (p, pe, s, se) | ||||
|      char *p, *pe, *s, *se; | ||||
| { | ||||
|   int ret; | ||||
|   char c1, c2; | ||||
| 
 | ||||
|   c1 = *pe; | ||||
|   c2 = *se; | ||||
| 
 | ||||
|   *pe = *se = '\0'; | ||||
| #if defined (HAVE_STRCOLL) | ||||
|   ret = strcoll (p, s); | ||||
| #else | ||||
|   ret = strcmp (p, s); | ||||
| #endif | ||||
| 
 | ||||
|   *pe = c1; | ||||
|   *se = c2; | ||||
| 
 | ||||
|   return (ret == 0 ? ret : FNM_NOMATCH); | ||||
| } | ||||
| 
 | ||||
| /* Match a ksh extended pattern specifier.  Return FNM_NOMATCH on failure or
 | ||||
|    0 on success.  This is handed the entire rest of the pattern and string | ||||
|    the first time an extended pattern specifier is encountered, so it calls | ||||
|    gmatch recursively. */ | ||||
| static int | ||||
| extmatch (xc, s, se, p, pe, flags) | ||||
|      int xc;		/* select which operation */ | ||||
|      char *s, *se; | ||||
|      char *p, *pe; | ||||
|      int flags; | ||||
| { | ||||
|   char *prest;			/* pointer to rest of pattern */ | ||||
|   char *psub;			/* pointer to sub-pattern */ | ||||
|   char *pnext;			/* pointer to next sub-pattern */ | ||||
|   char *srest;			/* pointer to rest of string */ | ||||
|   int m1, m2; | ||||
| 
 | ||||
| #if DEBUG_MATCHING | ||||
| fprintf(stderr, "extmatch: xc = %c\n", xc); | ||||
| fprintf(stderr, "extmatch: s = %s; se = %s\n", s, se); | ||||
| fprintf(stderr, "extmatch: p = %s; pe = %s\n", p, pe); | ||||
| #endif | ||||
| 
 | ||||
|   prest = patscan (p + (*p == '('), pe, 0); /* ) */ | ||||
|   if (prest == 0) | ||||
|     /* If PREST is 0, we failed to scan a valid pattern.  In this
 | ||||
|        case, we just want to compare the two as strings. */ | ||||
|     return (strcompare (p - 1, pe, s, se)); | ||||
| 
 | ||||
|   switch (xc) | ||||
|     { | ||||
|     case '+':			/* match one or more occurrences */ | ||||
|     case '*':			/* match zero or more occurrences */ | ||||
|       /* If we can get away with no matches, don't even bother.  Just
 | ||||
| 	 call gmatch on the rest of the pattern and return success if | ||||
| 	 it succeeds. */ | ||||
|       if (xc == '*' && (gmatch (s, se, prest, pe, flags) == 0)) | ||||
| 	return 0; | ||||
| 
 | ||||
|       /* OK, we have to do this the hard way.  First, we make sure one of
 | ||||
| 	 the subpatterns matches, then we try to match the rest of the | ||||
| 	 string. */ | ||||
|       for (psub = p + 1; ; psub = pnext) | ||||
| 	{ | ||||
| 	  pnext = patscan (psub, pe, '|'); | ||||
| 	  for (srest = s; srest <= se; srest++) | ||||
| 	    { | ||||
| 	      /* Match this substring (S -> SREST) against this
 | ||||
| 		 subpattern (psub -> pnext - 1) */ | ||||
| 	      m1 = gmatch (s, srest, psub, pnext - 1, flags) == 0; | ||||
| 	      /* OK, we matched a subpattern, so make sure the rest of the
 | ||||
| 		 string matches the rest of the pattern.  Also handle | ||||
| 		 multiple matches of the pattern. */ | ||||
| 	      if (m1) | ||||
| 		m2 = (gmatch (srest, se, prest, pe, flags) == 0) || | ||||
| 		      (s != srest && gmatch (srest, se, p - 1, pe, flags) == 0); | ||||
| 	      if (m1 && m2) | ||||
| 		return (0); | ||||
| 	    } | ||||
| 	  if (pnext == prest) | ||||
| 	    break; | ||||
| 	} | ||||
|       return (FNM_NOMATCH); | ||||
| 
 | ||||
|     case '?':		/* match zero or one of the patterns */ | ||||
|     case '@':		/* match exactly one of the patterns */ | ||||
|       /* If we can get away with no matches, don't even bother.  Just
 | ||||
| 	 call gmatch on the rest of the pattern and return success if | ||||
| 	 it succeeds. */ | ||||
|       if (xc == '?' && (gmatch (s, se, prest, pe, flags) == 0)) | ||||
| 	return 0; | ||||
| 
 | ||||
|       /* OK, we have to do this the hard way.  First, we see if one of
 | ||||
| 	 the subpatterns matches, then, if it does, we try to match the | ||||
| 	 rest of the string. */ | ||||
|       for (psub = p + 1; ; psub = pnext) | ||||
| 	{ | ||||
| 	  pnext = patscan (psub, pe, '|'); | ||||
| 	  srest = (prest == pe) ? se : s; | ||||
| 	  for ( ; srest <= se; srest++) | ||||
| 	    { | ||||
| 	      if (gmatch (s, srest, psub, pnext - 1, flags) == 0 && | ||||
| 		  gmatch (srest, se, prest, pe, flags) == 0) | ||||
| 		return (0); | ||||
| 	    } | ||||
| 	  if (pnext == prest) | ||||
| 	    break; | ||||
| 	} | ||||
|       return (FNM_NOMATCH); | ||||
| 
 | ||||
|     case '!':		/* match anything *except* one of the patterns */ | ||||
|       for (srest = s; srest <= se; srest++) | ||||
| 	{ | ||||
| 	  m1 = 0; | ||||
| 	  for (psub = p + 1; ; psub = pnext) | ||||
| 	    { | ||||
| 	      pnext = patscan (psub, pe, '|'); | ||||
| 	      /* If one of the patterns matches, just bail immediately. */ | ||||
| 	      if (m1 = (gmatch (s, srest, psub, pnext - 1, flags) == 0)) | ||||
| 		break; | ||||
| 	      if (pnext == prest) | ||||
| 		break; | ||||
| 	    } | ||||
| 	  if (m1 == 0 && gmatch (srest, se, prest, pe, flags) == 0) | ||||
| 	    return (0); | ||||
| 	} | ||||
|       return (FNM_NOMATCH); | ||||
|     } | ||||
| 
 | ||||
|   return (FNM_NOMATCH); | ||||
| } | ||||
| #endif /* EXTENDED_GLOB */ | ||||
| 
 | ||||
| #ifdef TEST | ||||
| main (c, v) | ||||
|  |  | |||
							
								
								
									
										116
									
								
								lib/glob/xmbsrtowcs.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										116
									
								
								lib/glob/xmbsrtowcs.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,116 @@ | |||
| /* xmbsrtowcs.c -- replacement function for mbsrtowcs */ | ||||
| 
 | ||||
| /* Copyright (C) 2002 Free Software Foundation, Inc.
 | ||||
| 
 | ||||
|    This file is part of GNU Bash, the Bourne Again SHell. | ||||
| 
 | ||||
|    Bash 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, or (at your option) any later | ||||
|    version. | ||||
| 
 | ||||
|    Bash 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 Bash; see the file COPYING.  If not, write to the Free Software | ||||
|    Foundation, 59 Temple Place, Suite 330, Boston, MA 02111 USA. */ | ||||
| #include <config.h> | ||||
| 
 | ||||
| #include <bashansi.h> | ||||
| 
 | ||||
| /* <wchar.h>, <wctype.h> and <stdlib.h> are included in "shmbutil.h".
 | ||||
|    If <wchar.h>, <wctype.h>, mbsrtowcs(), exist, HANDLE_MULTIBYTE | ||||
|    is defined as 1. */ | ||||
| #include <shmbutil.h> | ||||
| 
 | ||||
| #if HANDLE_MULTIBYTE | ||||
| /* On some locales (ex. ja_JP.sjis), mbsrtowc doesn't convert 0x5c to U<0x5c>.
 | ||||
|    So, this function is made for converting 0x5c to U<0x5c>. */ | ||||
| 
 | ||||
| static mbstate_t local_state; | ||||
| static int local_state_use = 0; | ||||
| 
 | ||||
| size_t | ||||
| xmbsrtowcs (dest, src, len, pstate) | ||||
|     wchar_t *dest; | ||||
|     const char **src; | ||||
|     size_t len; | ||||
|     mbstate_t *pstate; | ||||
| { | ||||
|   mbstate_t *ps; | ||||
|   size_t mblength, wclength, n; | ||||
| 
 | ||||
|   ps = pstate; | ||||
|   if (pstate == NULL) | ||||
|     { | ||||
|       if (!local_state_use) | ||||
| 	{ | ||||
| 	  memset (&local_state, '\0', sizeof(mbstate_t)); | ||||
| 	  local_state_use = 1; | ||||
| 	} | ||||
|       ps = &local_state; | ||||
|     } | ||||
| 
 | ||||
|   n = strlen(*src) + 1; | ||||
| 
 | ||||
|   if (dest == NULL) | ||||
|     { | ||||
|       wchar_t *wsbuf; | ||||
|       char *mbsbuf, *mbsbuf_top; | ||||
|       mbstate_t psbuf; | ||||
| 
 | ||||
|       wsbuf = (wchar_t *) malloc ((n + 1) * sizeof(wchar_t)); | ||||
|       mbsbuf_top = mbsbuf = (char *) malloc (n + 1); | ||||
|       memcpy(mbsbuf, *src, n + 1); | ||||
|       psbuf = *ps; | ||||
| 
 | ||||
|       wclength = mbsrtowcs (wsbuf, (const char **)&mbsbuf, n, &psbuf); | ||||
| 
 | ||||
|       free (wsbuf); | ||||
|       free (mbsbuf_top); | ||||
|       return wclength; | ||||
|     } | ||||
|        | ||||
|   for(wclength = 0; wclength < len; wclength++, dest++) | ||||
|     { | ||||
|       if(mbsinit(ps)) | ||||
| 	{ | ||||
| 	  if (**src == '\0') | ||||
| 	    { | ||||
| 	      *dest = L'\0'; | ||||
| 	      *src = NULL; | ||||
| 	      return (wclength); | ||||
| 	    } | ||||
| 	  else if (**src == '\\') | ||||
| 	    { | ||||
| 	      *dest = L'\\'; | ||||
| 	      mblength = 1; | ||||
| 	    } | ||||
| 	  else | ||||
| 	    mblength = mbrtowc(dest, *src, n, ps); | ||||
| 	} | ||||
|       else | ||||
| 	mblength = mbrtowc(dest, *src, n, ps); | ||||
| 
 | ||||
|       /* Cannot convert multibyte character to wide character. */ | ||||
|       if (mblength == (size_t)-1 || mblength == (size_t)-2) | ||||
| 	return (size_t)-1; | ||||
| 
 | ||||
|       *src += mblength; | ||||
|       n -= mblength; | ||||
| 
 | ||||
|       /* The multibyte string  has  been  completely  converted,
 | ||||
| 	 including  the terminating '\0'. */ | ||||
|       if (*dest == L'\0') | ||||
| 	{ | ||||
| 	  *src = NULL; | ||||
| 	  break; | ||||
| 	} | ||||
|     } | ||||
| 
 | ||||
|     return (wclength); | ||||
| } | ||||
| #endif /* HANDLE_MULTIBYTE */ | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Jari Aalto
				Jari Aalto