LCOV - code coverage report
Current view: top level - src/fe_utils - mbprint.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 135 176 76.7 %
Date: 2025-01-18 03:14:54 Functions: 7 8 87.5 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * Multibyte character printing support for frontend code
       4             :  *
       5             :  *
       6             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
       7             :  * Portions Copyright (c) 1994, Regents of the University of California
       8             :  *
       9             :  * src/fe_utils/mbprint.c
      10             :  *
      11             :  *-------------------------------------------------------------------------
      12             :  */
      13             : #include "postgres_fe.h"
      14             : 
      15             : #include "fe_utils/mbprint.h"
      16             : 
      17             : #include "libpq-fe.h"
      18             : 
      19             : 
      20             : /*
      21             :  * To avoid version-skew problems, this file must not use declarations
      22             :  * from pg_wchar.h: the encoding IDs we are dealing with are determined
      23             :  * by the libpq.so we are linked with, and that might not match the
      24             :  * numbers we see at compile time.  (If this file were inside libpq,
      25             :  * the problem would go away...)
      26             :  *
      27             :  * Hence, we have our own definition of pg_wchar, and we get the values
      28             :  * of any needed encoding IDs on-the-fly.
      29             :  */
      30             : 
      31             : typedef unsigned int pg_wchar;
      32             : 
      33             : static int
      34     4922664 : pg_get_utf8_id(void)
      35             : {
      36             :     static int  utf8_id = -1;
      37             : 
      38     4922664 :     if (utf8_id < 0)
      39       13192 :         utf8_id = pg_char_to_encoding("utf8");
      40     4922664 :     return utf8_id;
      41             : }
      42             : 
      43             : #define PG_UTF8     pg_get_utf8_id()
      44             : 
      45             : 
      46             : /*
      47             :  * Convert a UTF-8 character to a Unicode code point.
      48             :  * This is a one-character version of pg_utf2wchar_with_len.
      49             :  *
      50             :  * No error checks here, c must point to a long-enough string.
      51             :  */
      52             : static pg_wchar
      53           0 : utf8_to_unicode(const unsigned char *c)
      54             : {
      55           0 :     if ((*c & 0x80) == 0)
      56           0 :         return (pg_wchar) c[0];
      57           0 :     else if ((*c & 0xe0) == 0xc0)
      58           0 :         return (pg_wchar) (((c[0] & 0x1f) << 6) |
      59           0 :                            (c[1] & 0x3f));
      60           0 :     else if ((*c & 0xf0) == 0xe0)
      61           0 :         return (pg_wchar) (((c[0] & 0x0f) << 12) |
      62           0 :                            ((c[1] & 0x3f) << 6) |
      63           0 :                            (c[2] & 0x3f));
      64           0 :     else if ((*c & 0xf8) == 0xf0)
      65           0 :         return (pg_wchar) (((c[0] & 0x07) << 18) |
      66           0 :                            ((c[1] & 0x3f) << 12) |
      67           0 :                            ((c[2] & 0x3f) << 6) |
      68           0 :                            (c[3] & 0x3f));
      69             :     else
      70             :         /* that is an invalid code on purpose */
      71           0 :         return 0xffffffff;
      72             : }
      73             : 
      74             : 
      75             : /*
      76             :  * Unicode 3.1 compliant validation : for each category, it checks the
      77             :  * combination of each byte to make sure it maps to a valid range. It also
      78             :  * returns -1 for the following UCS values: ucs > 0x10ffff ucs & 0xfffe =
      79             :  * 0xfffe 0xfdd0 < ucs < 0xfdef ucs & 0xdb00 = 0xd800 (surrogates)
      80             :  */
      81             : static int
      82    18610282 : utf_charcheck(const unsigned char *c)
      83             : {
      84    18610282 :     if ((*c & 0x80) == 0)
      85    18608360 :         return 1;
      86        1922 :     else if ((*c & 0xe0) == 0xc0)
      87             :     {
      88             :         /* two-byte char */
      89        1640 :         if (((c[1] & 0xc0) == 0x80) && ((c[0] & 0x1f) > 0x01))
      90        1640 :             return 2;
      91           0 :         return -1;
      92             :     }
      93         282 :     else if ((*c & 0xf0) == 0xe0)
      94             :     {
      95             :         /* three-byte char */
      96         258 :         if (((c[1] & 0xc0) == 0x80) &&
      97         258 :             (((c[0] & 0x0f) != 0x00) || ((c[1] & 0x20) == 0x20)) &&
      98         258 :             ((c[2] & 0xc0) == 0x80))
      99             :         {
     100         258 :             int         z = c[0] & 0x0f;
     101         258 :             int         yx = ((c[1] & 0x3f) << 6) | (c[0] & 0x3f);
     102         258 :             int         lx = yx & 0x7f;
     103             : 
     104             :             /* check 0xfffe/0xffff, 0xfdd0..0xfedf range, surrogates */
     105         258 :             if (((z == 0x0f) &&
     106           0 :                  (((yx & 0xffe) == 0xffe) ||
     107         258 :                   (((yx & 0xf80) == 0xd80) && (lx >= 0x30) && (lx <= 0x4f)))) ||
     108           0 :                 ((z == 0x0d) && ((yx & 0xb00) == 0x800)))
     109           0 :                 return -1;
     110         258 :             return 3;
     111             :         }
     112           0 :         return -1;
     113             :     }
     114          24 :     else if ((*c & 0xf8) == 0xf0)
     115             :     {
     116          24 :         int         u = ((c[0] & 0x07) << 2) | ((c[1] & 0x30) >> 4);
     117             : 
     118             :         /* four-byte char */
     119          24 :         if (((c[1] & 0xc0) == 0x80) &&
     120          24 :             (u > 0x00) && (u <= 0x10) &&
     121          24 :             ((c[2] & 0xc0) == 0x80) && ((c[3] & 0xc0) == 0x80))
     122             :         {
     123             :             /* test for 0xzzzzfffe/0xzzzzfffff */
     124          24 :             if (((c[1] & 0x0f) == 0x0f) && ((c[2] & 0x3f) == 0x3f) &&
     125           0 :                 ((c[3] & 0x3e) == 0x3e))
     126           0 :                 return -1;
     127          24 :             return 4;
     128             :         }
     129           0 :         return -1;
     130             :     }
     131           0 :     return -1;
     132             : }
     133             : 
     134             : 
     135             : static void
     136     4919490 : mb_utf_validate(unsigned char *pwcs)
     137             : {
     138     4919490 :     unsigned char *p = pwcs;
     139             : 
     140    23529772 :     while (*pwcs)
     141             :     {
     142             :         int         len;
     143             : 
     144    18610282 :         if ((len = utf_charcheck(pwcs)) > 0)
     145             :         {
     146    18610282 :             if (p != pwcs)
     147             :             {
     148             :                 int         i;
     149             : 
     150           0 :                 for (i = 0; i < len; i++)
     151           0 :                     *p++ = *pwcs++;
     152             :             }
     153             :             else
     154             :             {
     155    18610282 :                 pwcs += len;
     156    18610282 :                 p += len;
     157             :             }
     158             :         }
     159             :         else
     160             :             /* we skip the char */
     161           0 :             pwcs++;
     162             :     }
     163     4919490 :     if (p != pwcs)
     164           0 :         *p = '\0';
     165     4919490 : }
     166             : 
     167             : /*
     168             :  * public functions : wcswidth and mbvalidate
     169             :  */
     170             : 
     171             : /*
     172             :  * pg_wcswidth is the dumb display-width function.
     173             :  * It assumes that everything will appear on one line.
     174             :  * OTOH it is easier to use than pg_wcssize if this applies to you.
     175             :  */
     176             : int
     177        4162 : pg_wcswidth(const char *pwcs, size_t len, int encoding)
     178             : {
     179        4162 :     int         width = 0;
     180             : 
     181       42266 :     while (len > 0)
     182             :     {
     183             :         int         chlen,
     184             :                     chwidth;
     185             : 
     186       38104 :         chlen = PQmblen(pwcs, encoding);
     187       38104 :         if (len < (size_t) chlen)
     188           0 :             break;              /* Invalid string */
     189             : 
     190       38104 :         chwidth = PQdsplen(pwcs, encoding);
     191       38104 :         if (chwidth > 0)
     192       38104 :             width += chwidth;
     193             : 
     194       38104 :         pwcs += chlen;
     195       38104 :         len -= chlen;
     196             :     }
     197        4162 :     return width;
     198             : }
     199             : 
     200             : /*
     201             :  * pg_wcssize takes the given string in the given encoding and returns three
     202             :  * values:
     203             :  *    result_width: Width in display characters of the longest line in string
     204             :  *    result_height: Number of lines in display output
     205             :  *    result_format_size: Number of bytes required to store formatted
     206             :  *      representation of string
     207             :  *
     208             :  * This MUST be kept in sync with pg_wcsformat!
     209             :  */
     210             : void
     211     2352260 : pg_wcssize(const unsigned char *pwcs, size_t len, int encoding,
     212             :            int *result_width, int *result_height, int *result_format_size)
     213             : {
     214             :     int         w,
     215     2352260 :                 chlen = 0,
     216     2352260 :                 linewidth = 0;
     217     2352260 :     int         width = 0;
     218     2352260 :     int         height = 1;
     219     2352260 :     int         format_size = 0;
     220             : 
     221    28244892 :     for (; *pwcs && len > 0; pwcs += chlen)
     222             :     {
     223    25892632 :         chlen = PQmblen((const char *) pwcs, encoding);
     224    25892632 :         if (len < (size_t) chlen)
     225           0 :             break;
     226    25892632 :         w = PQdsplen((const char *) pwcs, encoding);
     227             : 
     228    25892632 :         if (chlen == 1)         /* single-byte char */
     229             :         {
     230    25888788 :             if (*pwcs == '\n')  /* Newline */
     231             :             {
     232       37682 :                 if (linewidth > width)
     233        9020 :                     width = linewidth;
     234       37682 :                 linewidth = 0;
     235       37682 :                 height += 1;
     236       37682 :                 format_size += 1;   /* For NUL char */
     237             :             }
     238    25851106 :             else if (*pwcs == '\r') /* Linefeed */
     239             :             {
     240          16 :                 linewidth += 2;
     241          16 :                 format_size += 2;
     242             :             }
     243    25851090 :             else if (*pwcs == '\t') /* Tab */
     244             :             {
     245             :                 do
     246             :                 {
     247        3036 :                     linewidth++;
     248        3036 :                     format_size++;
     249        3036 :                 } while (linewidth % 8 != 0);
     250             :             }
     251    25850702 :             else if (w < 0)      /* Other control char */
     252             :             {
     253         144 :                 linewidth += 4;
     254         144 :                 format_size += 4;
     255             :             }
     256             :             else                /* Output it as-is */
     257             :             {
     258    25850558 :                 linewidth += w;
     259    25850558 :                 format_size += 1;
     260             :             }
     261             :         }
     262        3844 :         else if (w < 0)          /* Non-ascii control char */
     263             :         {
     264           0 :             linewidth += 6;     /* \u0000 */
     265           0 :             format_size += 6;
     266             :         }
     267             :         else                    /* All other chars */
     268             :         {
     269        3844 :             linewidth += w;
     270        3844 :             format_size += chlen;
     271             :         }
     272    25892632 :         len -= chlen;
     273             :     }
     274     2352260 :     if (linewidth > width)
     275     2174530 :         width = linewidth;
     276     2352260 :     format_size += 1;           /* For NUL char */
     277             : 
     278             :     /* Set results */
     279     2352260 :     if (result_width)
     280     2352260 :         *result_width = width;
     281     2352260 :     if (result_height)
     282     2352260 :         *result_height = height;
     283     2352260 :     if (result_format_size)
     284     2346724 :         *result_format_size = format_size;
     285     2352260 : }
     286             : 
     287             : /*
     288             :  *  Format a string into one or more "struct lineptr" lines.
     289             :  *  lines[i].ptr == NULL indicates the end of the array.
     290             :  *
     291             :  * This MUST be kept in sync with pg_wcssize!
     292             :  */
     293             : void
     294     1298890 : pg_wcsformat(const unsigned char *pwcs, size_t len, int encoding,
     295             :              struct lineptr *lines, int count)
     296             : {
     297             :     int         w,
     298     1298890 :                 chlen = 0;
     299     1298890 :     int         linewidth = 0;
     300     1298890 :     unsigned char *ptr = lines->ptr; /* Pointer to data area */
     301             : 
     302    15109456 :     for (; *pwcs && len > 0; pwcs += chlen)
     303             :     {
     304    13810566 :         chlen = PQmblen((const char *) pwcs, encoding);
     305    13810566 :         if (len < (size_t) chlen)
     306           0 :             break;
     307    13810566 :         w = PQdsplen((const char *) pwcs, encoding);
     308             : 
     309    13810566 :         if (chlen == 1)         /* single-byte char */
     310             :         {
     311    13808644 :             if (*pwcs == '\n')  /* Newline */
     312             :             {
     313       20074 :                 *ptr++ = '\0';
     314       20074 :                 lines->width = linewidth;
     315       20074 :                 linewidth = 0;
     316       20074 :                 lines++;
     317       20074 :                 count--;
     318       20074 :                 if (count <= 0)
     319           0 :                     exit(1);    /* Screwup */
     320             : 
     321             :                 /* make next line point to remaining memory */
     322       20074 :                 lines->ptr = ptr;
     323             :             }
     324    13788570 :             else if (*pwcs == '\r') /* Linefeed */
     325             :             {
     326           8 :                 strcpy((char *) ptr, "\\r");
     327           8 :                 linewidth += 2;
     328           8 :                 ptr += 2;
     329             :             }
     330    13788562 :             else if (*pwcs == '\t') /* Tab */
     331             :             {
     332             :                 do
     333             :                 {
     334        1518 :                     *ptr++ = ' ';
     335        1518 :                     linewidth++;
     336        1518 :                 } while (linewidth % 8 != 0);
     337             :             }
     338    13788368 :             else if (w < 0)      /* Other control char */
     339             :             {
     340          72 :                 sprintf((char *) ptr, "\\x%02X", *pwcs);
     341          72 :                 linewidth += 4;
     342          72 :                 ptr += 4;
     343             :             }
     344             :             else                /* Output it as-is */
     345             :             {
     346    13788296 :                 linewidth += w;
     347    13788296 :                 *ptr++ = *pwcs;
     348             :             }
     349             :         }
     350        1922 :         else if (w < 0)          /* Non-ascii control char */
     351             :         {
     352           0 :             if (encoding == PG_UTF8)
     353           0 :                 sprintf((char *) ptr, "\\u%04X", utf8_to_unicode(pwcs));
     354             :             else
     355             :             {
     356             :                 /*
     357             :                  * This case cannot happen in the current code because only
     358             :                  * UTF-8 signals multibyte control characters. But we may need
     359             :                  * to support it at some stage
     360             :                  */
     361           0 :                 sprintf((char *) ptr, "\\u????");
     362             :             }
     363           0 :             ptr += 6;
     364           0 :             linewidth += 6;
     365             :         }
     366             :         else                    /* All other chars */
     367             :         {
     368             :             int         i;
     369             : 
     370        6072 :             for (i = 0; i < chlen; i++)
     371        4150 :                 *ptr++ = pwcs[i];
     372        1922 :             linewidth += w;
     373             :         }
     374    13810566 :         len -= chlen;
     375             :     }
     376     1298890 :     lines->width = linewidth;
     377     1298890 :     *ptr++ = '\0';              /* Terminate formatted string */
     378             : 
     379     1298890 :     if (count <= 0)
     380           0 :         exit(1);                /* Screwup */
     381             : 
     382     1298890 :     (lines + 1)->ptr = NULL; /* terminate line array */
     383     1298890 : }
     384             : 
     385             : 
     386             : /*
     387             :  * Encoding validation: delete any unvalidatable characters from the string
     388             :  *
     389             :  * This seems redundant with existing functionality elsewhere?
     390             :  */
     391             : unsigned char *
     392     4922664 : mbvalidate(unsigned char *pwcs, int encoding)
     393             : {
     394     4922664 :     if (encoding == PG_UTF8)
     395     4919490 :         mb_utf_validate(pwcs);
     396             :     else
     397             :     {
     398             :         /*
     399             :          * other encodings needing validation should add their own routines
     400             :          * here
     401             :          */
     402             :     }
     403             : 
     404     4922664 :     return pwcs;
     405             : }

Generated by: LCOV version 1.14