⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 like.c

📁 PostgreSQL 8.1.4的源码 适用于Linux下的开源数据库系统
💻 C
字号:
/*------------------------------------------------------------------------- * * like.c *	  like expression handling code. * *	 NOTES *		A big hack of the regexp.c code!! Contributed by *		Keith Parks <emkxp01@mtcc.demon.co.uk> (7/95). * * Portions Copyright (c) 1996-2005, PostgreSQL Global Development Group * Portions Copyright (c) 1994, Regents of the University of California * * IDENTIFICATION *	$PostgreSQL: pgsql/src/backend/utils/adt/like.c,v 1.62 2005/10/15 02:49:28 momjian Exp $ * *------------------------------------------------------------------------- */#include "postgres.h"#include <ctype.h>#include "mb/pg_wchar.h"#include "utils/builtins.h"#define LIKE_TRUE						1#define LIKE_FALSE						0#define LIKE_ABORT						(-1)static int	MatchText(char *t, int tlen, char *p, int plen);static int	MatchTextIC(char *t, int tlen, char *p, int plen);static int	MatchBytea(char *t, int tlen, char *p, int plen);static text *do_like_escape(text *, text *);static int	MBMatchText(char *t, int tlen, char *p, int plen);static int	MBMatchTextIC(char *t, int tlen, char *p, int plen);static text *MB_do_like_escape(text *, text *);/*-------------------- * Support routine for MatchText. Compares given multibyte streams * as wide characters. If they match, returns 1 otherwise returns 0. *-------------------- */static intwchareq(char *p1, char *p2){	int			p1_len;	/* Optimization:  quickly compare the first byte. */	if (*p1 != *p2)		return (0);	p1_len = pg_mblen(p1);	if (pg_mblen(p2) != p1_len)		return (0);	/* They are the same length */	while (p1_len--)	{		if (*p1++ != *p2++)			return (0);	}	return (1);}/*-------------------- * Support routine for MatchTextIC. Compares given multibyte streams * as wide characters ignoring case. * If they match, returns 1 otherwise returns 0. *-------------------- */#define CHARMAX 0x80static intiwchareq(char *p1, char *p2){	pg_wchar	c1[2],				c2[2];	int			l;	/*	 * short cut. if *p1 and *p2 is lower than CHARMAX, then we could assume	 * they are ASCII	 */	if ((unsigned char) *p1 < CHARMAX && (unsigned char) *p2 < CHARMAX)		return (tolower((unsigned char) *p1) == tolower((unsigned char) *p2));	/*	 * if one of them is an ASCII while the other is not, then they must be	 * different characters	 */	else if ((unsigned char) *p1 < CHARMAX || (unsigned char) *p2 < CHARMAX)		return (0);	/*	 * ok, p1 and p2 are both > CHARMAX, then they must be multibyte	 * characters	 */	l = pg_mblen(p1);	(void) pg_mb2wchar_with_len(p1, c1, l);	c1[0] = tolower(c1[0]);	l = pg_mblen(p2);	(void) pg_mb2wchar_with_len(p2, c2, l);	c2[0] = tolower(c2[0]);	return (c1[0] == c2[0]);}#define CHAREQ(p1, p2) wchareq(p1, p2)#define ICHAREQ(p1, p2) iwchareq(p1, p2)#define NextChar(p, plen) \	do { int __l = pg_mblen(p); (p) +=__l; (plen) -=__l; } while (0)#define CopyAdvChar(dst, src, srclen) \	do { int __l = pg_mblen(src); \		 (srclen) -= __l; \		 while (__l-- > 0) \			 *(dst)++ = *(src)++; \	   } while (0)#define MatchText	MBMatchText#define MatchTextIC MBMatchTextIC#define do_like_escape	MB_do_like_escape#include "like_match.c"#undef CHAREQ#undef ICHAREQ#undef NextChar#undef CopyAdvChar#undef MatchText#undef MatchTextIC#undef do_like_escape#define CHAREQ(p1, p2) (*(p1) == *(p2))#define ICHAREQ(p1, p2) (tolower((unsigned char) *(p1)) == tolower((unsigned char) *(p2)))#define NextChar(p, plen) ((p)++, (plen)--)#define CopyAdvChar(dst, src, srclen) (*(dst)++ = *(src)++, (srclen)--)#define BYTEA_CHAREQ(p1, p2) (*(p1) == *(p2))#define BYTEA_NextChar(p, plen) ((p)++, (plen)--)#define BYTEA_CopyAdvChar(dst, src, srclen) (*(dst)++ = *(src)++, (srclen)--)#include "like_match.c"/* *	interface routines called by the function manager */Datumnamelike(PG_FUNCTION_ARGS){	Name		str = PG_GETARG_NAME(0);	text	   *pat = PG_GETARG_TEXT_P(1);	bool		result;	char	   *s,			   *p;	int			slen,				plen;	s = NameStr(*str);	slen = strlen(s);	p = VARDATA(pat);	plen = (VARSIZE(pat) - VARHDRSZ);	if (pg_database_encoding_max_length() == 1)		result = (MatchText(s, slen, p, plen) == LIKE_TRUE);	else		result = (MBMatchText(s, slen, p, plen) == LIKE_TRUE);	PG_RETURN_BOOL(result);}Datumnamenlike(PG_FUNCTION_ARGS){	Name		str = PG_GETARG_NAME(0);	text	   *pat = PG_GETARG_TEXT_P(1);	bool		result;	char	   *s,			   *p;	int			slen,				plen;	s = NameStr(*str);	slen = strlen(s);	p = VARDATA(pat);	plen = (VARSIZE(pat) - VARHDRSZ);	if (pg_database_encoding_max_length() == 1)		result = (MatchText(s, slen, p, plen) != LIKE_TRUE);	else		result = (MBMatchText(s, slen, p, plen) != LIKE_TRUE);	PG_RETURN_BOOL(result);}Datumtextlike(PG_FUNCTION_ARGS){	text	   *str = PG_GETARG_TEXT_P(0);	text	   *pat = PG_GETARG_TEXT_P(1);	bool		result;	char	   *s,			   *p;	int			slen,				plen;	s = VARDATA(str);	slen = (VARSIZE(str) - VARHDRSZ);	p = VARDATA(pat);	plen = (VARSIZE(pat) - VARHDRSZ);	if (pg_database_encoding_max_length() == 1)		result = (MatchText(s, slen, p, plen) == LIKE_TRUE);	else		result = (MBMatchText(s, slen, p, plen) == LIKE_TRUE);	PG_RETURN_BOOL(result);}Datumtextnlike(PG_FUNCTION_ARGS){	text	   *str = PG_GETARG_TEXT_P(0);	text	   *pat = PG_GETARG_TEXT_P(1);	bool		result;	char	   *s,			   *p;	int			slen,				plen;	s = VARDATA(str);	slen = (VARSIZE(str) - VARHDRSZ);	p = VARDATA(pat);	plen = (VARSIZE(pat) - VARHDRSZ);	if (pg_database_encoding_max_length() == 1)		result = (MatchText(s, slen, p, plen) != LIKE_TRUE);	else		result = (MBMatchText(s, slen, p, plen) != LIKE_TRUE);	PG_RETURN_BOOL(result);}Datumbytealike(PG_FUNCTION_ARGS){	bytea	   *str = PG_GETARG_BYTEA_P(0);	bytea	   *pat = PG_GETARG_BYTEA_P(1);	bool		result;	char	   *s,			   *p;	int			slen,				plen;	s = VARDATA(str);	slen = (VARSIZE(str) - VARHDRSZ);	p = VARDATA(pat);	plen = (VARSIZE(pat) - VARHDRSZ);	result = (MatchBytea(s, slen, p, plen) == LIKE_TRUE);	PG_RETURN_BOOL(result);}Datumbyteanlike(PG_FUNCTION_ARGS){	bytea	   *str = PG_GETARG_BYTEA_P(0);	bytea	   *pat = PG_GETARG_BYTEA_P(1);	bool		result;	char	   *s,			   *p;	int			slen,				plen;	s = VARDATA(str);	slen = (VARSIZE(str) - VARHDRSZ);	p = VARDATA(pat);	plen = (VARSIZE(pat) - VARHDRSZ);	result = (MatchBytea(s, slen, p, plen) != LIKE_TRUE);	PG_RETURN_BOOL(result);}/* * Case-insensitive versions */Datumnameiclike(PG_FUNCTION_ARGS){	Name		str = PG_GETARG_NAME(0);	text	   *pat = PG_GETARG_TEXT_P(1);	bool		result;	char	   *s,			   *p;	int			slen,				plen;	s = NameStr(*str);	slen = strlen(s);	p = VARDATA(pat);	plen = (VARSIZE(pat) - VARHDRSZ);	if (pg_database_encoding_max_length() == 1)		result = (MatchTextIC(s, slen, p, plen) == LIKE_TRUE);	else		result = (MBMatchTextIC(s, slen, p, plen) == LIKE_TRUE);	PG_RETURN_BOOL(result);}Datumnameicnlike(PG_FUNCTION_ARGS){	Name		str = PG_GETARG_NAME(0);	text	   *pat = PG_GETARG_TEXT_P(1);	bool		result;	char	   *s,			   *p;	int			slen,				plen;	s = NameStr(*str);	slen = strlen(s);	p = VARDATA(pat);	plen = (VARSIZE(pat) - VARHDRSZ);	if (pg_database_encoding_max_length() == 1)		result = (MatchTextIC(s, slen, p, plen) != LIKE_TRUE);	else		result = (MBMatchTextIC(s, slen, p, plen) != LIKE_TRUE);	PG_RETURN_BOOL(result);}Datumtexticlike(PG_FUNCTION_ARGS){	text	   *str = PG_GETARG_TEXT_P(0);	text	   *pat = PG_GETARG_TEXT_P(1);	bool		result;	char	   *s,			   *p;	int			slen,				plen;	s = VARDATA(str);	slen = (VARSIZE(str) - VARHDRSZ);	p = VARDATA(pat);	plen = (VARSIZE(pat) - VARHDRSZ);	if (pg_database_encoding_max_length() == 1)		result = (MatchTextIC(s, slen, p, plen) == LIKE_TRUE);	else		result = (MBMatchTextIC(s, slen, p, plen) == LIKE_TRUE);	PG_RETURN_BOOL(result);}Datumtexticnlike(PG_FUNCTION_ARGS){	text	   *str = PG_GETARG_TEXT_P(0);	text	   *pat = PG_GETARG_TEXT_P(1);	bool		result;	char	   *s,			   *p;	int			slen,				plen;	s = VARDATA(str);	slen = (VARSIZE(str) - VARHDRSZ);	p = VARDATA(pat);	plen = (VARSIZE(pat) - VARHDRSZ);	if (pg_database_encoding_max_length() == 1)		result = (MatchTextIC(s, slen, p, plen) != LIKE_TRUE);	else		result = (MBMatchTextIC(s, slen, p, plen) != LIKE_TRUE);	PG_RETURN_BOOL(result);}/* * like_escape() --- given a pattern and an ESCAPE string, * convert the pattern to use Postgres' standard backslash escape convention. */Datumlike_escape(PG_FUNCTION_ARGS){	text	   *pat = PG_GETARG_TEXT_P(0);	text	   *esc = PG_GETARG_TEXT_P(1);	text	   *result;	if (pg_database_encoding_max_length() == 1)		result = do_like_escape(pat, esc);	else		result = MB_do_like_escape(pat, esc);	PG_RETURN_TEXT_P(result);}/* * like_escape_bytea() --- given a pattern and an ESCAPE string, * convert the pattern to use Postgres' standard backslash escape convention. */Datumlike_escape_bytea(PG_FUNCTION_ARGS){	bytea	   *pat = PG_GETARG_BYTEA_P(0);	bytea	   *esc = PG_GETARG_BYTEA_P(1);	bytea	   *result;	char	   *p,			   *e,			   *r;	int			plen,				elen;	bool		afterescape;	p = VARDATA(pat);	plen = (VARSIZE(pat) - VARHDRSZ);	e = VARDATA(esc);	elen = (VARSIZE(esc) - VARHDRSZ);	/*	 * Worst-case pattern growth is 2x --- unlikely, but it's hardly worth	 * trying to calculate the size more accurately than that.	 */	result = (text *) palloc(plen * 2 + VARHDRSZ);	r = VARDATA(result);	if (elen == 0)	{		/*		 * No escape character is wanted.  Double any backslashes in the		 * pattern to make them act like ordinary characters.		 */		while (plen > 0)		{			if (*p == '\\')				*r++ = '\\';			BYTEA_CopyAdvChar(r, p, plen);		}	}	else	{		/*		 * The specified escape must be only a single character.		 */		BYTEA_NextChar(e, elen);		if (elen != 0)			ereport(ERROR,					(errcode(ERRCODE_INVALID_ESCAPE_SEQUENCE),					 errmsg("invalid escape string"),				  errhint("Escape string must be empty or one character.")));		e = VARDATA(esc);		/*		 * If specified escape is '\', just copy the pattern as-is.		 */		if (*e == '\\')		{			memcpy(result, pat, VARSIZE(pat));			PG_RETURN_BYTEA_P(result);		}		/*		 * Otherwise, convert occurrences of the specified escape character to		 * '\', and double occurrences of '\' --- unless they immediately		 * follow an escape character!		 */		afterescape = false;		while (plen > 0)		{			if (BYTEA_CHAREQ(p, e) && !afterescape)			{				*r++ = '\\';				BYTEA_NextChar(p, plen);				afterescape = true;			}			else if (*p == '\\')			{				*r++ = '\\';				if (!afterescape)					*r++ = '\\';				BYTEA_NextChar(p, plen);				afterescape = false;			}			else			{				BYTEA_CopyAdvChar(r, p, plen);				afterescape = false;			}		}	}	VARATT_SIZEP(result) = r - ((char *) result);	PG_RETURN_BYTEA_P(result);}/* * Same as above, but specifically for bytea (binary) datatype */static intMatchBytea(char *t, int tlen, char *p, int plen){	/* Fast path for match-everything pattern */	if ((plen == 1) && (*p == '%'))		return LIKE_TRUE;	while ((tlen > 0) && (plen > 0))	{		if (*p == '\\')		{			/* Next pattern char must match literally, whatever it is */			BYTEA_NextChar(p, plen);			if ((plen <= 0) || !BYTEA_CHAREQ(t, p))				return LIKE_FALSE;		}		else if (*p == '%')		{			/* %% is the same as % according to the SQL standard */			/* Advance past all %'s */			while ((plen > 0) && (*p == '%'))				BYTEA_NextChar(p, plen);			/* Trailing percent matches everything. */			if (plen <= 0)				return LIKE_TRUE;			/*			 * Otherwise, scan for a text position at which we can match the			 * rest of the pattern.			 */			while (tlen > 0)			{				/*				 * Optimization to prevent most recursion: don't recurse				 * unless first pattern char might match this text char.				 */				if (BYTEA_CHAREQ(t, p) || (*p == '\\') || (*p == '_'))				{					int			matched = MatchBytea(t, tlen, p, plen);					if (matched != LIKE_FALSE)						return matched; /* TRUE or ABORT */				}				BYTEA_NextChar(t, tlen);			}			/*			 * End of text with no match, so no point in trying later places			 * to start matching this pattern.			 */			return LIKE_ABORT;		}		else if ((*p != '_') && !BYTEA_CHAREQ(t, p))		{			/*			 * Not the single-character wildcard and no explicit match? Then			 * time to quit...			 */			return LIKE_FALSE;		}		BYTEA_NextChar(t, tlen);		BYTEA_NextChar(p, plen);	}	if (tlen > 0)		return LIKE_FALSE;		/* end of pattern, but not of text */	/* End of input string.  Do we have matching pattern remaining? */	while ((plen > 0) && (*p == '%'))	/* allow multiple %'s at end of										 * pattern */		BYTEA_NextChar(p, plen);	if (plen <= 0)		return LIKE_TRUE;	/*	 * End of text with no match, so no point in trying later places to start	 * matching this pattern.	 */	return LIKE_ABORT;}	/* MatchBytea() */

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -