LCOV - code coverage report
Current view: top level - src/backend/utils/adt - oracle_compat.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 336 393 85.5 %
Date: 2025-01-18 04:15:08 Functions: 20 20 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  * oracle_compat.c
       3             :  *  Oracle compatible functions.
       4             :  *
       5             :  * Copyright (c) 1996-2025, PostgreSQL Global Development Group
       6             :  *
       7             :  *  Author: Edmund Mergl <E.Mergl@bawue.de>
       8             :  *  Multibyte enhancement: Tatsuo Ishii <ishii@postgresql.org>
       9             :  *
      10             :  *
      11             :  * IDENTIFICATION
      12             :  *  src/backend/utils/adt/oracle_compat.c
      13             :  *
      14             :  *-------------------------------------------------------------------------
      15             :  */
      16             : #include "postgres.h"
      17             : 
      18             : #include "common/int.h"
      19             : #include "mb/pg_wchar.h"
      20             : #include "miscadmin.h"
      21             : #include "utils/builtins.h"
      22             : #include "utils/formatting.h"
      23             : #include "utils/memutils.h"
      24             : #include "varatt.h"
      25             : 
      26             : 
      27             : static text *dotrim(const char *string, int stringlen,
      28             :                     const char *set, int setlen,
      29             :                     bool doltrim, bool dortrim);
      30             : static bytea *dobyteatrim(bytea *string, bytea *set,
      31             :                           bool doltrim, bool dortrim);
      32             : 
      33             : 
      34             : /********************************************************************
      35             :  *
      36             :  * lower
      37             :  *
      38             :  * Syntax:
      39             :  *
      40             :  *   text lower(text string)
      41             :  *
      42             :  * Purpose:
      43             :  *
      44             :  *   Returns string, with all letters forced to lowercase.
      45             :  *
      46             :  ********************************************************************/
      47             : 
      48             : Datum
      49      182902 : lower(PG_FUNCTION_ARGS)
      50             : {
      51      182902 :     text       *in_string = PG_GETARG_TEXT_PP(0);
      52             :     char       *out_string;
      53             :     text       *result;
      54             : 
      55      182902 :     out_string = str_tolower(VARDATA_ANY(in_string),
      56      182902 :                              VARSIZE_ANY_EXHDR(in_string),
      57             :                              PG_GET_COLLATION());
      58      182902 :     result = cstring_to_text(out_string);
      59      182902 :     pfree(out_string);
      60             : 
      61      182902 :     PG_RETURN_TEXT_P(result);
      62             : }
      63             : 
      64             : 
      65             : /********************************************************************
      66             :  *
      67             :  * upper
      68             :  *
      69             :  * Syntax:
      70             :  *
      71             :  *   text upper(text string)
      72             :  *
      73             :  * Purpose:
      74             :  *
      75             :  *   Returns string, with all letters forced to uppercase.
      76             :  *
      77             :  ********************************************************************/
      78             : 
      79             : Datum
      80     1049278 : upper(PG_FUNCTION_ARGS)
      81             : {
      82     1049278 :     text       *in_string = PG_GETARG_TEXT_PP(0);
      83             :     char       *out_string;
      84             :     text       *result;
      85             : 
      86     1049278 :     out_string = str_toupper(VARDATA_ANY(in_string),
      87     1049278 :                              VARSIZE_ANY_EXHDR(in_string),
      88             :                              PG_GET_COLLATION());
      89     1049278 :     result = cstring_to_text(out_string);
      90     1049278 :     pfree(out_string);
      91             : 
      92     1049278 :     PG_RETURN_TEXT_P(result);
      93             : }
      94             : 
      95             : 
      96             : /********************************************************************
      97             :  *
      98             :  * initcap
      99             :  *
     100             :  * Syntax:
     101             :  *
     102             :  *   text initcap(text string)
     103             :  *
     104             :  * Purpose:
     105             :  *
     106             :  *   Returns string, with first letter of each word in uppercase, all
     107             :  *   other letters in lowercase. A word is defined as a sequence of
     108             :  *   alphanumeric characters, delimited by non-alphanumeric
     109             :  *   characters.
     110             :  *
     111             :  ********************************************************************/
     112             : 
     113             : Datum
     114         202 : initcap(PG_FUNCTION_ARGS)
     115             : {
     116         202 :     text       *in_string = PG_GETARG_TEXT_PP(0);
     117             :     char       *out_string;
     118             :     text       *result;
     119             : 
     120         202 :     out_string = str_initcap(VARDATA_ANY(in_string),
     121         202 :                              VARSIZE_ANY_EXHDR(in_string),
     122             :                              PG_GET_COLLATION());
     123         202 :     result = cstring_to_text(out_string);
     124         202 :     pfree(out_string);
     125             : 
     126         202 :     PG_RETURN_TEXT_P(result);
     127             : }
     128             : 
     129             : 
     130             : /********************************************************************
     131             :  *
     132             :  * lpad
     133             :  *
     134             :  * Syntax:
     135             :  *
     136             :  *   text lpad(text string1, int4 len, text string2)
     137             :  *
     138             :  * Purpose:
     139             :  *
     140             :  *   Returns string1, left-padded to length len with the sequence of
     141             :  *   characters in string2.  If len is less than the length of string1,
     142             :  *   instead truncate (on the right) to len.
     143             :  *
     144             :  ********************************************************************/
     145             : 
     146             : Datum
     147       32770 : lpad(PG_FUNCTION_ARGS)
     148             : {
     149       32770 :     text       *string1 = PG_GETARG_TEXT_PP(0);
     150       32770 :     int32       len = PG_GETARG_INT32(1);
     151       32770 :     text       *string2 = PG_GETARG_TEXT_PP(2);
     152             :     text       *ret;
     153             :     char       *ptr1,
     154             :                *ptr2,
     155             :                *ptr2start,
     156             :                *ptr2end,
     157             :                *ptr_ret;
     158             :     int         m,
     159             :                 s1len,
     160             :                 s2len;
     161             :     int         bytelen;
     162             : 
     163             :     /* Negative len is silently taken as zero */
     164       32770 :     if (len < 0)
     165           6 :         len = 0;
     166             : 
     167       32770 :     s1len = VARSIZE_ANY_EXHDR(string1);
     168       32770 :     if (s1len < 0)
     169           0 :         s1len = 0;              /* shouldn't happen */
     170             : 
     171       32770 :     s2len = VARSIZE_ANY_EXHDR(string2);
     172       32770 :     if (s2len < 0)
     173           0 :         s2len = 0;              /* shouldn't happen */
     174             : 
     175       32770 :     s1len = pg_mbstrlen_with_len(VARDATA_ANY(string1), s1len);
     176             : 
     177       32770 :     if (s1len > len)
     178          70 :         s1len = len;            /* truncate string1 to len chars */
     179             : 
     180       32770 :     if (s2len <= 0)
     181           6 :         len = s1len;            /* nothing to pad with, so don't pad */
     182             : 
     183             :     /* compute worst-case output length */
     184       32770 :     if (unlikely(pg_mul_s32_overflow(pg_database_encoding_max_length(), len,
     185       32770 :                                      &bytelen)) ||
     186       32770 :         unlikely(pg_add_s32_overflow(bytelen, VARHDRSZ, &bytelen)) ||
     187       32770 :         unlikely(!AllocSizeIsValid(bytelen)))
     188           0 :         ereport(ERROR,
     189             :                 (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
     190             :                  errmsg("requested length too large")));
     191             : 
     192       32770 :     ret = (text *) palloc(bytelen);
     193             : 
     194       32770 :     m = len - s1len;
     195             : 
     196       32770 :     ptr2 = ptr2start = VARDATA_ANY(string2);
     197       32770 :     ptr2end = ptr2 + s2len;
     198       32770 :     ptr_ret = VARDATA(ret);
     199             : 
     200       34886 :     while (m--)
     201             :     {
     202        2116 :         int         mlen = pg_mblen(ptr2);
     203             : 
     204        2116 :         memcpy(ptr_ret, ptr2, mlen);
     205        2116 :         ptr_ret += mlen;
     206        2116 :         ptr2 += mlen;
     207        2116 :         if (ptr2 == ptr2end)    /* wrap around at end of s2 */
     208        2092 :             ptr2 = ptr2start;
     209             :     }
     210             : 
     211       32770 :     ptr1 = VARDATA_ANY(string1);
     212             : 
     213       96242 :     while (s1len--)
     214             :     {
     215       63472 :         int         mlen = pg_mblen(ptr1);
     216             : 
     217       63472 :         memcpy(ptr_ret, ptr1, mlen);
     218       63472 :         ptr_ret += mlen;
     219       63472 :         ptr1 += mlen;
     220             :     }
     221             : 
     222       32770 :     SET_VARSIZE(ret, ptr_ret - (char *) ret);
     223             : 
     224       32770 :     PG_RETURN_TEXT_P(ret);
     225             : }
     226             : 
     227             : 
     228             : /********************************************************************
     229             :  *
     230             :  * rpad
     231             :  *
     232             :  * Syntax:
     233             :  *
     234             :  *   text rpad(text string1, int4 len, text string2)
     235             :  *
     236             :  * Purpose:
     237             :  *
     238             :  *   Returns string1, right-padded to length len with the sequence of
     239             :  *   characters in string2.  If len is less than the length of string1,
     240             :  *   instead truncate (on the right) to len.
     241             :  *
     242             :  ********************************************************************/
     243             : 
     244             : Datum
     245          44 : rpad(PG_FUNCTION_ARGS)
     246             : {
     247          44 :     text       *string1 = PG_GETARG_TEXT_PP(0);
     248          44 :     int32       len = PG_GETARG_INT32(1);
     249          44 :     text       *string2 = PG_GETARG_TEXT_PP(2);
     250             :     text       *ret;
     251             :     char       *ptr1,
     252             :                *ptr2,
     253             :                *ptr2start,
     254             :                *ptr2end,
     255             :                *ptr_ret;
     256             :     int         m,
     257             :                 s1len,
     258             :                 s2len;
     259             :     int         bytelen;
     260             : 
     261             :     /* Negative len is silently taken as zero */
     262          44 :     if (len < 0)
     263           6 :         len = 0;
     264             : 
     265          44 :     s1len = VARSIZE_ANY_EXHDR(string1);
     266          44 :     if (s1len < 0)
     267           0 :         s1len = 0;              /* shouldn't happen */
     268             : 
     269          44 :     s2len = VARSIZE_ANY_EXHDR(string2);
     270          44 :     if (s2len < 0)
     271           0 :         s2len = 0;              /* shouldn't happen */
     272             : 
     273          44 :     s1len = pg_mbstrlen_with_len(VARDATA_ANY(string1), s1len);
     274             : 
     275          44 :     if (s1len > len)
     276          12 :         s1len = len;            /* truncate string1 to len chars */
     277             : 
     278          44 :     if (s2len <= 0)
     279           6 :         len = s1len;            /* nothing to pad with, so don't pad */
     280             : 
     281             :     /* compute worst-case output length */
     282          44 :     if (unlikely(pg_mul_s32_overflow(pg_database_encoding_max_length(), len,
     283          44 :                                      &bytelen)) ||
     284          44 :         unlikely(pg_add_s32_overflow(bytelen, VARHDRSZ, &bytelen)) ||
     285          44 :         unlikely(!AllocSizeIsValid(bytelen)))
     286           0 :         ereport(ERROR,
     287             :                 (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
     288             :                  errmsg("requested length too large")));
     289             : 
     290          44 :     ret = (text *) palloc(bytelen);
     291             : 
     292          44 :     m = len - s1len;
     293             : 
     294          44 :     ptr1 = VARDATA_ANY(string1);
     295          44 :     ptr_ret = VARDATA(ret);
     296             : 
     297         108 :     while (s1len--)
     298             :     {
     299          64 :         int         mlen = pg_mblen(ptr1);
     300             : 
     301          64 :         memcpy(ptr_ret, ptr1, mlen);
     302          64 :         ptr_ret += mlen;
     303          64 :         ptr1 += mlen;
     304             :     }
     305             : 
     306          44 :     ptr2 = ptr2start = VARDATA_ANY(string2);
     307          44 :     ptr2end = ptr2 + s2len;
     308             : 
     309     1920104 :     while (m--)
     310             :     {
     311     1920060 :         int         mlen = pg_mblen(ptr2);
     312             : 
     313     1920060 :         memcpy(ptr_ret, ptr2, mlen);
     314     1920060 :         ptr_ret += mlen;
     315     1920060 :         ptr2 += mlen;
     316     1920060 :         if (ptr2 == ptr2end)    /* wrap around at end of s2 */
     317     1920036 :             ptr2 = ptr2start;
     318             :     }
     319             : 
     320          44 :     SET_VARSIZE(ret, ptr_ret - (char *) ret);
     321             : 
     322          44 :     PG_RETURN_TEXT_P(ret);
     323             : }
     324             : 
     325             : 
     326             : /********************************************************************
     327             :  *
     328             :  * btrim
     329             :  *
     330             :  * Syntax:
     331             :  *
     332             :  *   text btrim(text string, text set)
     333             :  *
     334             :  * Purpose:
     335             :  *
     336             :  *   Returns string with characters removed from the front and back
     337             :  *   up to the first character not in set.
     338             :  *
     339             :  ********************************************************************/
     340             : 
     341             : Datum
     342          20 : btrim(PG_FUNCTION_ARGS)
     343             : {
     344          20 :     text       *string = PG_GETARG_TEXT_PP(0);
     345          20 :     text       *set = PG_GETARG_TEXT_PP(1);
     346             :     text       *ret;
     347             : 
     348          40 :     ret = dotrim(VARDATA_ANY(string), VARSIZE_ANY_EXHDR(string),
     349          40 :                  VARDATA_ANY(set), VARSIZE_ANY_EXHDR(set),
     350             :                  true, true);
     351             : 
     352          20 :     PG_RETURN_TEXT_P(ret);
     353             : }
     354             : 
     355             : /********************************************************************
     356             :  *
     357             :  * btrim1 --- btrim with set fixed as ' '
     358             :  *
     359             :  ********************************************************************/
     360             : 
     361             : Datum
     362         562 : btrim1(PG_FUNCTION_ARGS)
     363             : {
     364         562 :     text       *string = PG_GETARG_TEXT_PP(0);
     365             :     text       *ret;
     366             : 
     367         562 :     ret = dotrim(VARDATA_ANY(string), VARSIZE_ANY_EXHDR(string),
     368             :                  " ", 1,
     369             :                  true, true);
     370             : 
     371         562 :     PG_RETURN_TEXT_P(ret);
     372             : }
     373             : 
     374             : /*
     375             :  * Common implementation for btrim, ltrim, rtrim
     376             :  */
     377             : static text *
     378       34298 : dotrim(const char *string, int stringlen,
     379             :        const char *set, int setlen,
     380             :        bool doltrim, bool dortrim)
     381             : {
     382             :     int         i;
     383             : 
     384             :     /* Nothing to do if either string or set is empty */
     385       34298 :     if (stringlen > 0 && setlen > 0)
     386             :     {
     387       34298 :         if (pg_database_encoding_max_length() > 1)
     388             :         {
     389             :             /*
     390             :              * In the multibyte-encoding case, build arrays of pointers to
     391             :              * character starts, so that we can avoid inefficient checks in
     392             :              * the inner loops.
     393             :              */
     394             :             const char **stringchars;
     395             :             const char **setchars;
     396             :             int        *stringmblen;
     397             :             int        *setmblen;
     398             :             int         stringnchars;
     399             :             int         setnchars;
     400             :             int         resultndx;
     401             :             int         resultnchars;
     402             :             const char *p;
     403             :             int         len;
     404             :             int         mblen;
     405             :             const char *str_pos;
     406             :             int         str_len;
     407             : 
     408       34276 :             stringchars = (const char **) palloc(stringlen * sizeof(char *));
     409       34276 :             stringmblen = (int *) palloc(stringlen * sizeof(int));
     410       34276 :             stringnchars = 0;
     411       34276 :             p = string;
     412       34276 :             len = stringlen;
     413      304588 :             while (len > 0)
     414             :             {
     415      270312 :                 stringchars[stringnchars] = p;
     416      270312 :                 stringmblen[stringnchars] = mblen = pg_mblen(p);
     417      270312 :                 stringnchars++;
     418      270312 :                 p += mblen;
     419      270312 :                 len -= mblen;
     420             :             }
     421             : 
     422       34276 :             setchars = (const char **) palloc(setlen * sizeof(char *));
     423       34276 :             setmblen = (int *) palloc(setlen * sizeof(int));
     424       34276 :             setnchars = 0;
     425       34276 :             p = set;
     426       34276 :             len = setlen;
     427       69230 :             while (len > 0)
     428             :             {
     429       34954 :                 setchars[setnchars] = p;
     430       34954 :                 setmblen[setnchars] = mblen = pg_mblen(p);
     431       34954 :                 setnchars++;
     432       34954 :                 p += mblen;
     433       34954 :                 len -= mblen;
     434             :             }
     435             : 
     436       34276 :             resultndx = 0;      /* index in stringchars[] */
     437       34276 :             resultnchars = stringnchars;
     438             : 
     439       34276 :             if (doltrim)
     440             :             {
     441       56588 :                 while (resultnchars > 0)
     442             :                 {
     443       56588 :                     str_pos = stringchars[resultndx];
     444       56588 :                     str_len = stringmblen[resultndx];
     445       85210 :                     for (i = 0; i < setnchars; i++)
     446             :                     {
     447       56708 :                         if (str_len == setmblen[i] &&
     448       56708 :                             memcmp(str_pos, setchars[i], str_len) == 0)
     449       28086 :                             break;
     450             :                     }
     451       56588 :                     if (i >= setnchars)
     452       28502 :                         break;  /* no match here */
     453       28086 :                     string += str_len;
     454       28086 :                     stringlen -= str_len;
     455       28086 :                     resultndx++;
     456       28086 :                     resultnchars--;
     457             :                 }
     458             :             }
     459             : 
     460       34276 :             if (dortrim)
     461             :             {
     462       74544 :                 while (resultnchars > 0)
     463             :                 {
     464       74544 :                     str_pos = stringchars[resultndx + resultnchars - 1];
     465       74544 :                     str_len = stringmblen[resultndx + resultnchars - 1];
     466       82670 :                     for (i = 0; i < setnchars; i++)
     467             :                     {
     468       76314 :                         if (str_len == setmblen[i] &&
     469       76314 :                             memcmp(str_pos, setchars[i], str_len) == 0)
     470       68188 :                             break;
     471             :                     }
     472       74544 :                     if (i >= setnchars)
     473        6356 :                         break;  /* no match here */
     474       68188 :                     stringlen -= str_len;
     475       68188 :                     resultnchars--;
     476             :                 }
     477             :             }
     478             : 
     479       34276 :             pfree(stringchars);
     480       34276 :             pfree(stringmblen);
     481       34276 :             pfree(setchars);
     482       34276 :             pfree(setmblen);
     483             :         }
     484             :         else
     485             :         {
     486             :             /*
     487             :              * In the single-byte-encoding case, we don't need such overhead.
     488             :              */
     489          22 :             if (doltrim)
     490             :             {
     491           0 :                 while (stringlen > 0)
     492             :                 {
     493           0 :                     char        str_ch = *string;
     494             : 
     495           0 :                     for (i = 0; i < setlen; i++)
     496             :                     {
     497           0 :                         if (str_ch == set[i])
     498           0 :                             break;
     499             :                     }
     500           0 :                     if (i >= setlen)
     501           0 :                         break;  /* no match here */
     502           0 :                     string++;
     503           0 :                     stringlen--;
     504             :                 }
     505             :             }
     506             : 
     507          22 :             if (dortrim)
     508             :             {
     509          44 :                 while (stringlen > 0)
     510             :                 {
     511          44 :                     char        str_ch = string[stringlen - 1];
     512             : 
     513          66 :                     for (i = 0; i < setlen; i++)
     514             :                     {
     515          44 :                         if (str_ch == set[i])
     516          22 :                             break;
     517             :                     }
     518          44 :                     if (i >= setlen)
     519          22 :                         break;  /* no match here */
     520          22 :                     stringlen--;
     521             :                 }
     522             :             }
     523             :         }
     524             :     }
     525             : 
     526             :     /* Return selected portion of string */
     527       34298 :     return cstring_to_text_with_len(string, stringlen);
     528             : }
     529             : 
     530             : /*
     531             :  * Common implementation for bytea versions of btrim, ltrim, rtrim
     532             :  */
     533             : bytea *
     534          36 : dobyteatrim(bytea *string, bytea *set, bool doltrim, bool dortrim)
     535             : {
     536             :     bytea      *ret;
     537             :     char       *ptr,
     538             :                *end,
     539             :                *ptr2,
     540             :                *ptr2start,
     541             :                *end2;
     542             :     int         m,
     543             :                 stringlen,
     544             :                 setlen;
     545             : 
     546          36 :     stringlen = VARSIZE_ANY_EXHDR(string);
     547          36 :     setlen = VARSIZE_ANY_EXHDR(set);
     548             : 
     549          36 :     if (stringlen <= 0 || setlen <= 0)
     550          12 :         return string;
     551             : 
     552          24 :     m = stringlen;
     553          24 :     ptr = VARDATA_ANY(string);
     554          24 :     end = ptr + stringlen - 1;
     555          24 :     ptr2start = VARDATA_ANY(set);
     556          24 :     end2 = ptr2start + setlen - 1;
     557             : 
     558          24 :     if (doltrim)
     559             :     {
     560          36 :         while (m > 0)
     561             :         {
     562          36 :             ptr2 = ptr2start;
     563          54 :             while (ptr2 <= end2)
     564             :             {
     565          36 :                 if (*ptr == *ptr2)
     566          18 :                     break;
     567          18 :                 ++ptr2;
     568             :             }
     569          36 :             if (ptr2 > end2)
     570          18 :                 break;
     571          18 :             ptr++;
     572          18 :             m--;
     573             :         }
     574             :     }
     575             : 
     576          24 :     if (dortrim)
     577             :     {
     578          36 :         while (m > 0)
     579             :         {
     580          36 :             ptr2 = ptr2start;
     581          54 :             while (ptr2 <= end2)
     582             :             {
     583          36 :                 if (*end == *ptr2)
     584          18 :                     break;
     585          18 :                 ++ptr2;
     586             :             }
     587          36 :             if (ptr2 > end2)
     588          18 :                 break;
     589          18 :             end--;
     590          18 :             m--;
     591             :         }
     592             :     }
     593             : 
     594          24 :     ret = (bytea *) palloc(VARHDRSZ + m);
     595          24 :     SET_VARSIZE(ret, VARHDRSZ + m);
     596          24 :     memcpy(VARDATA(ret), ptr, m);
     597          24 :     return ret;
     598             : }
     599             : 
     600             : /********************************************************************
     601             :  *
     602             :  * byteatrim
     603             :  *
     604             :  * Syntax:
     605             :  *
     606             :  *   bytea byteatrim(bytea string, bytea set)
     607             :  *
     608             :  * Purpose:
     609             :  *
     610             :  *   Returns string with characters removed from the front and back
     611             :  *   up to the first character not in set.
     612             :  *
     613             :  * Cloned from btrim and modified as required.
     614             :  ********************************************************************/
     615             : 
     616             : Datum
     617          24 : byteatrim(PG_FUNCTION_ARGS)
     618             : {
     619          24 :     bytea      *string = PG_GETARG_BYTEA_PP(0);
     620          24 :     bytea      *set = PG_GETARG_BYTEA_PP(1);
     621             :     bytea      *ret;
     622             : 
     623          24 :     ret = dobyteatrim(string, set, true, true);
     624             : 
     625          24 :     PG_RETURN_BYTEA_P(ret);
     626             : }
     627             : 
     628             : /********************************************************************
     629             :  *
     630             :  * bytealtrim
     631             :  *
     632             :  * Syntax:
     633             :  *
     634             :  *   bytea bytealtrim(bytea string, bytea set)
     635             :  *
     636             :  * Purpose:
     637             :  *
     638             :  *   Returns string with initial characters removed up to the first
     639             :  *   character not in set.
     640             :  *
     641             :  ********************************************************************/
     642             : 
     643             : Datum
     644           6 : bytealtrim(PG_FUNCTION_ARGS)
     645             : {
     646           6 :     bytea      *string = PG_GETARG_BYTEA_PP(0);
     647           6 :     bytea      *set = PG_GETARG_BYTEA_PP(1);
     648             :     bytea      *ret;
     649             : 
     650           6 :     ret = dobyteatrim(string, set, true, false);
     651             : 
     652           6 :     PG_RETURN_BYTEA_P(ret);
     653             : }
     654             : 
     655             : /********************************************************************
     656             :  *
     657             :  * byteartrim
     658             :  *
     659             :  * Syntax:
     660             :  *
     661             :  *   bytea byteartrim(bytea string, bytea set)
     662             :  *
     663             :  * Purpose:
     664             :  *
     665             :  *   Returns string with final characters removed after the last
     666             :  *   character not in set.
     667             :  *
     668             :  ********************************************************************/
     669             : 
     670             : Datum
     671           6 : byteartrim(PG_FUNCTION_ARGS)
     672             : {
     673           6 :     bytea      *string = PG_GETARG_BYTEA_PP(0);
     674           6 :     bytea      *set = PG_GETARG_BYTEA_PP(1);
     675             :     bytea      *ret;
     676             : 
     677           6 :     ret = dobyteatrim(string, set, false, true);
     678             : 
     679           6 :     PG_RETURN_BYTEA_P(ret);
     680             : }
     681             : 
     682             : /********************************************************************
     683             :  *
     684             :  * ltrim
     685             :  *
     686             :  * Syntax:
     687             :  *
     688             :  *   text ltrim(text string, text set)
     689             :  *
     690             :  * Purpose:
     691             :  *
     692             :  *   Returns string with initial characters removed up to the first
     693             :  *   character not in set.
     694             :  *
     695             :  ********************************************************************/
     696             : 
     697             : Datum
     698       27912 : ltrim(PG_FUNCTION_ARGS)
     699             : {
     700       27912 :     text       *string = PG_GETARG_TEXT_PP(0);
     701       27912 :     text       *set = PG_GETARG_TEXT_PP(1);
     702             :     text       *ret;
     703             : 
     704       55824 :     ret = dotrim(VARDATA_ANY(string), VARSIZE_ANY_EXHDR(string),
     705       55824 :                  VARDATA_ANY(set), VARSIZE_ANY_EXHDR(set),
     706             :                  true, false);
     707             : 
     708       27912 :     PG_RETURN_TEXT_P(ret);
     709             : }
     710             : 
     711             : /********************************************************************
     712             :  *
     713             :  * ltrim1 --- ltrim with set fixed as ' '
     714             :  *
     715             :  ********************************************************************/
     716             : 
     717             : Datum
     718           8 : ltrim1(PG_FUNCTION_ARGS)
     719             : {
     720           8 :     text       *string = PG_GETARG_TEXT_PP(0);
     721             :     text       *ret;
     722             : 
     723           8 :     ret = dotrim(VARDATA_ANY(string), VARSIZE_ANY_EXHDR(string),
     724             :                  " ", 1,
     725             :                  true, false);
     726             : 
     727           8 :     PG_RETURN_TEXT_P(ret);
     728             : }
     729             : 
     730             : /********************************************************************
     731             :  *
     732             :  * rtrim
     733             :  *
     734             :  * Syntax:
     735             :  *
     736             :  *   text rtrim(text string, text set)
     737             :  *
     738             :  * Purpose:
     739             :  *
     740             :  *   Returns string with final characters removed after the last
     741             :  *   character not in set.
     742             :  *
     743             :  ********************************************************************/
     744             : 
     745             : Datum
     746         272 : rtrim(PG_FUNCTION_ARGS)
     747             : {
     748         272 :     text       *string = PG_GETARG_TEXT_PP(0);
     749         272 :     text       *set = PG_GETARG_TEXT_PP(1);
     750             :     text       *ret;
     751             : 
     752         544 :     ret = dotrim(VARDATA_ANY(string), VARSIZE_ANY_EXHDR(string),
     753         544 :                  VARDATA_ANY(set), VARSIZE_ANY_EXHDR(set),
     754             :                  false, true);
     755             : 
     756         272 :     PG_RETURN_TEXT_P(ret);
     757             : }
     758             : 
     759             : /********************************************************************
     760             :  *
     761             :  * rtrim1 --- rtrim with set fixed as ' '
     762             :  *
     763             :  ********************************************************************/
     764             : 
     765             : Datum
     766        5524 : rtrim1(PG_FUNCTION_ARGS)
     767             : {
     768        5524 :     text       *string = PG_GETARG_TEXT_PP(0);
     769             :     text       *ret;
     770             : 
     771        5524 :     ret = dotrim(VARDATA_ANY(string), VARSIZE_ANY_EXHDR(string),
     772             :                  " ", 1,
     773             :                  false, true);
     774             : 
     775        5524 :     PG_RETURN_TEXT_P(ret);
     776             : }
     777             : 
     778             : 
     779             : /********************************************************************
     780             :  *
     781             :  * translate
     782             :  *
     783             :  * Syntax:
     784             :  *
     785             :  *   text translate(text string, text from, text to)
     786             :  *
     787             :  * Purpose:
     788             :  *
     789             :  *   Returns string after replacing all occurrences of characters in from
     790             :  *   with the corresponding character in to.  If from is longer than to,
     791             :  *   occurrences of the extra characters in from are deleted.
     792             :  *   Improved by Edwin Ramirez <ramirez@doc.mssm.edu>.
     793             :  *
     794             :  ********************************************************************/
     795             : 
     796             : Datum
     797          34 : translate(PG_FUNCTION_ARGS)
     798             : {
     799          34 :     text       *string = PG_GETARG_TEXT_PP(0);
     800          34 :     text       *from = PG_GETARG_TEXT_PP(1);
     801          34 :     text       *to = PG_GETARG_TEXT_PP(2);
     802             :     text       *result;
     803             :     char       *from_ptr,
     804             :                *to_ptr,
     805             :                *to_end;
     806             :     char       *source,
     807             :                *target;
     808             :     int         m,
     809             :                 fromlen,
     810             :                 tolen,
     811             :                 retlen,
     812             :                 i;
     813             :     int         bytelen;
     814             :     int         len;
     815             :     int         source_len;
     816             :     int         from_index;
     817             : 
     818          34 :     m = VARSIZE_ANY_EXHDR(string);
     819          34 :     if (m <= 0)
     820           6 :         PG_RETURN_TEXT_P(string);
     821          28 :     source = VARDATA_ANY(string);
     822             : 
     823          28 :     fromlen = VARSIZE_ANY_EXHDR(from);
     824          28 :     from_ptr = VARDATA_ANY(from);
     825          28 :     tolen = VARSIZE_ANY_EXHDR(to);
     826          28 :     to_ptr = VARDATA_ANY(to);
     827          28 :     to_end = to_ptr + tolen;
     828             : 
     829             :     /*
     830             :      * The worst-case expansion is to substitute a max-length character for a
     831             :      * single-byte character at each position of the string.
     832             :      */
     833          28 :     if (unlikely(pg_mul_s32_overflow(pg_database_encoding_max_length(), m,
     834          28 :                                      &bytelen)) ||
     835          28 :         unlikely(pg_add_s32_overflow(bytelen, VARHDRSZ, &bytelen)) ||
     836          28 :         unlikely(!AllocSizeIsValid(bytelen)))
     837           0 :         ereport(ERROR,
     838             :                 (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
     839             :                  errmsg("requested length too large")));
     840             : 
     841          28 :     result = (text *) palloc(bytelen);
     842             : 
     843          28 :     target = VARDATA(result);
     844          28 :     retlen = 0;
     845             : 
     846         280 :     while (m > 0)
     847             :     {
     848         252 :         source_len = pg_mblen(source);
     849         252 :         from_index = 0;
     850             : 
     851         684 :         for (i = 0; i < fromlen; i += len)
     852             :         {
     853         494 :             len = pg_mblen(&from_ptr[i]);
     854         494 :             if (len == source_len &&
     855         494 :                 memcmp(source, &from_ptr[i], len) == 0)
     856          62 :                 break;
     857             : 
     858         432 :             from_index++;
     859             :         }
     860         252 :         if (i < fromlen)
     861             :         {
     862             :             /* substitute, or delete if no corresponding "to" character */
     863          62 :             char       *p = to_ptr;
     864             : 
     865          96 :             for (i = 0; i < from_index; i++)
     866             :             {
     867          40 :                 if (p >= to_end)
     868           6 :                     break;
     869          34 :                 p += pg_mblen(p);
     870             :             }
     871          62 :             if (p < to_end)
     872             :             {
     873          50 :                 len = pg_mblen(p);
     874          50 :                 memcpy(target, p, len);
     875          50 :                 target += len;
     876          50 :                 retlen += len;
     877             :             }
     878             :         }
     879             :         else
     880             :         {
     881             :             /* no match, so copy */
     882         190 :             memcpy(target, source, source_len);
     883         190 :             target += source_len;
     884         190 :             retlen += source_len;
     885             :         }
     886             : 
     887         252 :         source += source_len;
     888         252 :         m -= source_len;
     889             :     }
     890             : 
     891          28 :     SET_VARSIZE(result, retlen + VARHDRSZ);
     892             : 
     893             :     /*
     894             :      * The function result is probably much bigger than needed, if we're using
     895             :      * a multibyte encoding, but it's not worth reallocating it; the result
     896             :      * probably won't live long anyway.
     897             :      */
     898             : 
     899          28 :     PG_RETURN_TEXT_P(result);
     900             : }
     901             : 
     902             : /********************************************************************
     903             :  *
     904             :  * ascii
     905             :  *
     906             :  * Syntax:
     907             :  *
     908             :  *   int ascii(text string)
     909             :  *
     910             :  * Purpose:
     911             :  *
     912             :  *   Returns the decimal representation of the first character from
     913             :  *   string.
     914             :  *   If the string is empty we return 0.
     915             :  *   If the database encoding is UTF8, we return the Unicode codepoint.
     916             :  *   If the database encoding is any other multi-byte encoding, we
     917             :  *   return the value of the first byte if it is an ASCII character
     918             :  *   (range 1 .. 127), or raise an error.
     919             :  *   For all other encodings we return the value of the first byte,
     920             :  *   (range 1..255).
     921             :  *
     922             :  ********************************************************************/
     923             : 
     924             : Datum
     925          58 : ascii(PG_FUNCTION_ARGS)
     926             : {
     927          58 :     text       *string = PG_GETARG_TEXT_PP(0);
     928          58 :     int         encoding = GetDatabaseEncoding();
     929             :     unsigned char *data;
     930             : 
     931          58 :     if (VARSIZE_ANY_EXHDR(string) <= 0)
     932           6 :         PG_RETURN_INT32(0);
     933             : 
     934          52 :     data = (unsigned char *) VARDATA_ANY(string);
     935             : 
     936          52 :     if (encoding == PG_UTF8 && *data > 127)
     937             :     {
     938             :         /* return the code point for Unicode */
     939             : 
     940           0 :         int         result = 0,
     941           0 :                     tbytes = 0,
     942             :                     i;
     943             : 
     944           0 :         if (*data >= 0xF0)
     945             :         {
     946           0 :             result = *data & 0x07;
     947           0 :             tbytes = 3;
     948             :         }
     949           0 :         else if (*data >= 0xE0)
     950             :         {
     951           0 :             result = *data & 0x0F;
     952           0 :             tbytes = 2;
     953             :         }
     954             :         else
     955             :         {
     956             :             Assert(*data > 0xC0);
     957           0 :             result = *data & 0x1f;
     958           0 :             tbytes = 1;
     959             :         }
     960             : 
     961             :         Assert(tbytes > 0);
     962             : 
     963           0 :         for (i = 1; i <= tbytes; i++)
     964             :         {
     965             :             Assert((data[i] & 0xC0) == 0x80);
     966           0 :             result = (result << 6) + (data[i] & 0x3f);
     967             :         }
     968             : 
     969           0 :         PG_RETURN_INT32(result);
     970             :     }
     971             :     else
     972             :     {
     973          52 :         if (pg_encoding_max_length(encoding) > 1 && *data > 127)
     974           0 :             ereport(ERROR,
     975             :                     (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
     976             :                      errmsg("requested character too large")));
     977             : 
     978             : 
     979          52 :         PG_RETURN_INT32((int32) *data);
     980             :     }
     981             : }
     982             : 
     983             : /********************************************************************
     984             :  *
     985             :  * chr
     986             :  *
     987             :  * Syntax:
     988             :  *
     989             :  *   text chr(int val)
     990             :  *
     991             :  * Purpose:
     992             :  *
     993             :  *  Returns the character having the binary equivalent to val.
     994             :  *
     995             :  * For UTF8 we treat the argument as a Unicode code point.
     996             :  * For other multi-byte encodings we raise an error for arguments
     997             :  * outside the strict ASCII range (1..127).
     998             :  *
     999             :  * It's important that we don't ever return a value that is not valid
    1000             :  * in the database encoding, so that this doesn't become a way for
    1001             :  * invalid data to enter the database.
    1002             :  *
    1003             :  ********************************************************************/
    1004             : 
    1005             : Datum
    1006      544868 : chr         (PG_FUNCTION_ARGS)
    1007             : {
    1008      544868 :     int32       arg = PG_GETARG_INT32(0);
    1009             :     uint32      cvalue;
    1010             :     text       *result;
    1011      544868 :     int         encoding = GetDatabaseEncoding();
    1012             : 
    1013             :     /*
    1014             :      * Error out on arguments that make no sense or that we can't validly
    1015             :      * represent in the encoding.
    1016             :      */
    1017      544868 :     if (arg < 0)
    1018           0 :         ereport(ERROR,
    1019             :                 (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
    1020             :                  errmsg("character number must be positive")));
    1021      544868 :     else if (arg == 0)
    1022           6 :         ereport(ERROR,
    1023             :                 (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
    1024             :                  errmsg("null character not permitted")));
    1025             : 
    1026      544862 :     cvalue = arg;
    1027             : 
    1028      544862 :     if (encoding == PG_UTF8 && cvalue > 127)
    1029           0 :     {
    1030             :         /* for Unicode we treat the argument as a code point */
    1031             :         int         bytes;
    1032             :         unsigned char *wch;
    1033             : 
    1034             :         /*
    1035             :          * We only allow valid Unicode code points; per RFC3629 that stops at
    1036             :          * U+10FFFF, even though 4-byte UTF8 sequences can hold values up to
    1037             :          * U+1FFFFF.
    1038             :          */
    1039           0 :         if (cvalue > 0x0010ffff)
    1040           0 :             ereport(ERROR,
    1041             :                     (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
    1042             :                      errmsg("requested character too large for encoding: %u",
    1043             :                             cvalue)));
    1044             : 
    1045           0 :         if (cvalue > 0xffff)
    1046           0 :             bytes = 4;
    1047           0 :         else if (cvalue > 0x07ff)
    1048           0 :             bytes = 3;
    1049             :         else
    1050           0 :             bytes = 2;
    1051             : 
    1052           0 :         result = (text *) palloc(VARHDRSZ + bytes);
    1053           0 :         SET_VARSIZE(result, VARHDRSZ + bytes);
    1054           0 :         wch = (unsigned char *) VARDATA(result);
    1055             : 
    1056           0 :         if (bytes == 2)
    1057             :         {
    1058           0 :             wch[0] = 0xC0 | ((cvalue >> 6) & 0x1F);
    1059           0 :             wch[1] = 0x80 | (cvalue & 0x3F);
    1060             :         }
    1061           0 :         else if (bytes == 3)
    1062             :         {
    1063           0 :             wch[0] = 0xE0 | ((cvalue >> 12) & 0x0F);
    1064           0 :             wch[1] = 0x80 | ((cvalue >> 6) & 0x3F);
    1065           0 :             wch[2] = 0x80 | (cvalue & 0x3F);
    1066             :         }
    1067             :         else
    1068             :         {
    1069           0 :             wch[0] = 0xF0 | ((cvalue >> 18) & 0x07);
    1070           0 :             wch[1] = 0x80 | ((cvalue >> 12) & 0x3F);
    1071           0 :             wch[2] = 0x80 | ((cvalue >> 6) & 0x3F);
    1072           0 :             wch[3] = 0x80 | (cvalue & 0x3F);
    1073             :         }
    1074             : 
    1075             :         /*
    1076             :          * The preceding range check isn't sufficient, because UTF8 excludes
    1077             :          * Unicode "surrogate pair" codes.  Make sure what we created is valid
    1078             :          * UTF8.
    1079             :          */
    1080           0 :         if (!pg_utf8_islegal(wch, bytes))
    1081           0 :             ereport(ERROR,
    1082             :                     (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
    1083             :                      errmsg("requested character not valid for encoding: %u",
    1084             :                             cvalue)));
    1085             :     }
    1086             :     else
    1087             :     {
    1088             :         bool        is_mb;
    1089             : 
    1090      544862 :         is_mb = pg_encoding_max_length(encoding) > 1;
    1091             : 
    1092      544862 :         if ((is_mb && (cvalue > 127)) || (!is_mb && (cvalue > 255)))
    1093           0 :             ereport(ERROR,
    1094             :                     (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
    1095             :                      errmsg("requested character too large for encoding: %u",
    1096             :                             cvalue)));
    1097             : 
    1098      544862 :         result = (text *) palloc(VARHDRSZ + 1);
    1099      544862 :         SET_VARSIZE(result, VARHDRSZ + 1);
    1100      544862 :         *VARDATA(result) = (char) cvalue;
    1101             :     }
    1102             : 
    1103      544862 :     PG_RETURN_TEXT_P(result);
    1104             : }
    1105             : 
    1106             : /********************************************************************
    1107             :  *
    1108             :  * repeat
    1109             :  *
    1110             :  * Syntax:
    1111             :  *
    1112             :  *   text repeat(text string, int val)
    1113             :  *
    1114             :  * Purpose:
    1115             :  *
    1116             :  *  Repeat string by val.
    1117             :  *
    1118             :  ********************************************************************/
    1119             : 
    1120             : Datum
    1121      230582 : repeat(PG_FUNCTION_ARGS)
    1122             : {
    1123      230582 :     text       *string = PG_GETARG_TEXT_PP(0);
    1124      230582 :     int32       count = PG_GETARG_INT32(1);
    1125             :     text       *result;
    1126             :     int         slen,
    1127             :                 tlen;
    1128             :     int         i;
    1129             :     char       *cp,
    1130             :                *sp;
    1131             : 
    1132      230582 :     if (count < 0)
    1133           6 :         count = 0;
    1134             : 
    1135      230582 :     slen = VARSIZE_ANY_EXHDR(string);
    1136             : 
    1137      230582 :     if (unlikely(pg_mul_s32_overflow(count, slen, &tlen)) ||
    1138      230582 :         unlikely(pg_add_s32_overflow(tlen, VARHDRSZ, &tlen)) ||
    1139      230582 :         unlikely(!AllocSizeIsValid(tlen)))
    1140           0 :         ereport(ERROR,
    1141             :                 (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
    1142             :                  errmsg("requested length too large")));
    1143             : 
    1144      230582 :     result = (text *) palloc(tlen);
    1145             : 
    1146      230582 :     SET_VARSIZE(result, tlen);
    1147      230582 :     cp = VARDATA(result);
    1148      230582 :     sp = VARDATA_ANY(string);
    1149    49869496 :     for (i = 0; i < count; i++)
    1150             :     {
    1151    49638914 :         memcpy(cp, sp, slen);
    1152    49638914 :         cp += slen;
    1153    49638914 :         CHECK_FOR_INTERRUPTS();
    1154             :     }
    1155             : 
    1156      230582 :     PG_RETURN_TEXT_P(result);
    1157             : }

Generated by: LCOV version 1.14