mirror of
https://github.com/postgres/postgres.git
synced 2025-10-24 01:29:19 +03:00
Fix bogus calculation of potential output string length in translate().
This commit is contained in:
@@ -9,7 +9,7 @@
|
|||||||
*
|
*
|
||||||
*
|
*
|
||||||
* IDENTIFICATION
|
* IDENTIFICATION
|
||||||
* $PostgreSQL: pgsql/src/backend/utils/adt/oracle_compat.c,v 1.67.2.1 2007/02/08 20:33:54 momjian Exp $
|
* $PostgreSQL: pgsql/src/backend/utils/adt/oracle_compat.c,v 1.67.2.2 2007/09/22 05:35:52 tgl Exp $
|
||||||
*
|
*
|
||||||
*-------------------------------------------------------------------------
|
*-------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
@@ -1153,27 +1153,34 @@ translate(PG_FUNCTION_ARGS)
|
|||||||
tolen,
|
tolen,
|
||||||
retlen,
|
retlen,
|
||||||
i;
|
i;
|
||||||
|
int worst_len;
|
||||||
int str_len;
|
|
||||||
int estimate_len;
|
|
||||||
int len;
|
int len;
|
||||||
int source_len;
|
int source_len;
|
||||||
int from_index;
|
int from_index;
|
||||||
|
|
||||||
if ((m = VARSIZE(string) - VARHDRSZ) <= 0)
|
m = VARSIZE(string) - VARHDRSZ;
|
||||||
|
if (m <= 0)
|
||||||
PG_RETURN_TEXT_P(string);
|
PG_RETURN_TEXT_P(string);
|
||||||
|
source = VARDATA(string);
|
||||||
|
|
||||||
fromlen = VARSIZE(from) - VARHDRSZ;
|
fromlen = VARSIZE(from) - VARHDRSZ;
|
||||||
from_ptr = VARDATA(from);
|
from_ptr = VARDATA(from);
|
||||||
tolen = VARSIZE(to) - VARHDRSZ;
|
tolen = VARSIZE(to) - VARHDRSZ;
|
||||||
to_ptr = VARDATA(to);
|
to_ptr = VARDATA(to);
|
||||||
|
|
||||||
str_len = VARSIZE(string);
|
/*
|
||||||
estimate_len = (tolen * 1.0 / fromlen + 0.5) * str_len;
|
* The worst-case expansion is to substitute a max-length character for
|
||||||
estimate_len = estimate_len > str_len ? estimate_len : str_len;
|
* a single-byte character at each position of the string.
|
||||||
result = (text *) palloc(estimate_len);
|
*/
|
||||||
|
worst_len = pg_database_encoding_max_length() * m;
|
||||||
|
|
||||||
source = VARDATA(string);
|
/* check for integer overflow */
|
||||||
|
if (worst_len / pg_database_encoding_max_length() != m)
|
||||||
|
ereport(ERROR,
|
||||||
|
(errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
|
||||||
|
errmsg("requested length too large")));
|
||||||
|
|
||||||
|
result = (text *) palloc(worst_len + VARHDRSZ);
|
||||||
target = VARDATA(result);
|
target = VARDATA(result);
|
||||||
retlen = 0;
|
retlen = 0;
|
||||||
|
|
||||||
@@ -1226,9 +1233,9 @@ translate(PG_FUNCTION_ARGS)
|
|||||||
VARATT_SIZEP(result) = retlen + VARHDRSZ;
|
VARATT_SIZEP(result) = retlen + VARHDRSZ;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* There may be some wasted space in the result if deletions occurred, but
|
* The function result is probably much bigger than needed, if we're
|
||||||
* it's not worth reallocating it; the function result probably won't live
|
* using a multibyte encoding, but it's not worth reallocating it;
|
||||||
* long anyway.
|
* the result probably won't live long anyway.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
PG_RETURN_TEXT_P(result);
|
PG_RETURN_TEXT_P(result);
|
||||||
|
Reference in New Issue
Block a user