Ensure maxlen is at leat 1 in dict_int
authorTomas Vondra <tomas.vondra@postgresql.org>
Tue, 3 Dec 2019 15:55:51 +0000 (16:55 +0100)
committerTomas Vondra <tomas.vondra@postgresql.org>
Tue, 3 Dec 2019 17:41:29 +0000 (18:41 +0100)
The dict_int text search dictionary template accepts maxlen parameter,
which is then used to cap the length of input strings. The value was
not properly checked, and the code simply does

    txt[d->maxlen] = '\0';

to insert a terminator, leading to segfaults with negative values.

This commit simply rejects values less than 1. The issue was there since
dct_int was introduced in 9.3, so backpatch all the way back to 9.4
which is the oldest supported version.

Reported-by: cili
Discussion: https://postgr.es/m/16144-a36a5bef7657047d@postgresql.org
Backpatch-through: 9.4

contrib/dict_int/dict_int.c
contrib/dict_int/expected/dict_int.out
contrib/dict_int/sql/dict_int.sql

index 56ede37089eacc233fd087ed8ac31f26e523d89e..13b5696d83c060ca084fd5aceb1ff0e98fd5af7b 100644 (file)
@@ -45,6 +45,11 @@ dintdict_init(PG_FUNCTION_ARGS)
        if (strcmp(defel->defname, "maxlen") == 0)
        {
            d->maxlen = atoi(defGetString(defel));
+
+           if (d->maxlen < 1)
+               ereport(ERROR,
+                       (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
+                        errmsg("maxlen value has to be >= 1")));
        }
        else if (strcmp(defel->defname, "rejectlong") == 0)
        {
index 3b766ec52ade4616d613b77fa646a285ab1221a7..483e700d23153826260941955409c00c4961f22e 100644 (file)
@@ -300,3 +300,5 @@ select ts_lexize('intdict', '314532610153');
  {314532}
 (1 row)
 
+ALTER TEXT SEARCH DICTIONARY intdict (MAXLEN = -214783648);
+ERROR:  maxlen value has to be >= 1
index 8ffec6b77089a45984068d7f38f693f450eefa90..5c27accff4a4ab9fb034c1f08d7db9c8c8b9362b 100644 (file)
@@ -51,3 +51,5 @@ select ts_lexize('intdict', '252281774');
 select ts_lexize('intdict', '313425');
 select ts_lexize('intdict', '641439323669');
 select ts_lexize('intdict', '314532610153');
+
+ALTER TEXT SEARCH DICTIONARY intdict (MAXLEN = -214783648);