Fix buffer overrun when performing repr() on a unicode string in a build
with wide unicode (UCS-4) support. Obtained from: Python SVN (#51466) Security: CVE-2006-4980 With hat: secteam
This commit is contained in:
parent
fa6c54db2a
commit
a1560e82eb
Notes:
svn2git
2021-03-31 03:12:20 +00:00
svn path=/head/; revision=174893
4 changed files with 136 additions and 0 deletions
|
@ -7,6 +7,7 @@
|
|||
|
||||
PORTNAME= python
|
||||
PORTVERSION= 2.4.3
|
||||
PORTREVISION= 1
|
||||
CATEGORIES= lang python ipv6
|
||||
MASTER_SITES= ${PYTHON_MASTER_SITES}
|
||||
MASTER_SITE_SUBDIR= ${PYTHON_MASTER_SITE_SUBDIR}
|
||||
|
|
67
lang/python/files/patch-CVE-2006-4980
Normal file
67
lang/python/files/patch-CVE-2006-4980
Normal file
|
@ -0,0 +1,67 @@
|
|||
--- Objects/unicodeobject.c 2006/08/22 08:09:11 51465
|
||||
+++ Objects/unicodeobject.c 2006/08/22 08:25:33 51466
|
||||
@@ -1970,7 +1970,28 @@
|
||||
|
||||
static const char *hexdigit = "0123456789abcdef";
|
||||
|
||||
- repr = PyString_FromStringAndSize(NULL, 2 + 6*size + 1);
|
||||
+ /* Initial allocation is based on the longest-possible unichr
|
||||
+ escape.
|
||||
+
|
||||
+ In wide (UTF-32) builds '\U00xxxxxx' is 10 chars per source
|
||||
+ unichr, so in this case it's the longest unichr escape. In
|
||||
+ narrow (UTF-16) builds this is five chars per source unichr
|
||||
+ since there are two unichrs in the surrogate pair, so in narrow
|
||||
+ (UTF-16) builds it's not the longest unichr escape.
|
||||
+
|
||||
+ In wide or narrow builds '\uxxxx' is 6 chars per source unichr,
|
||||
+ so in the narrow (UTF-16) build case it's the longest unichr
|
||||
+ escape.
|
||||
+ */
|
||||
+
|
||||
+ repr = PyString_FromStringAndSize(NULL,
|
||||
+ 2
|
||||
+#ifdef Py_UNICODE_WIDE
|
||||
+ + 10*size
|
||||
+#else
|
||||
+ + 6*size
|
||||
+#endif
|
||||
+ + 1);
|
||||
if (repr == NULL)
|
||||
return NULL;
|
||||
|
||||
@@ -1995,15 +2016,6 @@
|
||||
#ifdef Py_UNICODE_WIDE
|
||||
/* Map 21-bit characters to '\U00xxxxxx' */
|
||||
else if (ch >= 0x10000) {
|
||||
- int offset = p - PyString_AS_STRING(repr);
|
||||
-
|
||||
- /* Resize the string if necessary */
|
||||
- if (offset + 12 > PyString_GET_SIZE(repr)) {
|
||||
- if (_PyString_Resize(&repr, PyString_GET_SIZE(repr) + 100))
|
||||
- return NULL;
|
||||
- p = PyString_AS_STRING(repr) + offset;
|
||||
- }
|
||||
-
|
||||
*p++ = '\\';
|
||||
*p++ = 'U';
|
||||
*p++ = hexdigit[(ch >> 28) & 0x0000000F];
|
||||
@@ -2016,8 +2028,8 @@
|
||||
*p++ = hexdigit[ch & 0x0000000F];
|
||||
continue;
|
||||
}
|
||||
-#endif
|
||||
- /* Map UTF-16 surrogate pairs to Unicode \UXXXXXXXX escapes */
|
||||
+#else
|
||||
+ /* Map UTF-16 surrogate pairs to '\U00xxxxxx' */
|
||||
else if (ch >= 0xD800 && ch < 0xDC00) {
|
||||
Py_UNICODE ch2;
|
||||
Py_UCS4 ucs;
|
||||
@@ -2042,6 +2054,7 @@
|
||||
s--;
|
||||
size++;
|
||||
}
|
||||
+#endif
|
||||
|
||||
/* Map 16-bit characters to '\uxxxx' */
|
||||
if (ch >= 256) {
|
|
@ -7,6 +7,7 @@
|
|||
|
||||
PORTNAME= python
|
||||
PORTVERSION= 2.4.3
|
||||
PORTREVISION= 1
|
||||
CATEGORIES= lang python ipv6
|
||||
MASTER_SITES= ${PYTHON_MASTER_SITES}
|
||||
MASTER_SITE_SUBDIR= ${PYTHON_MASTER_SITE_SUBDIR}
|
||||
|
|
67
lang/python24/files/patch-CVE-2006-4980
Normal file
67
lang/python24/files/patch-CVE-2006-4980
Normal file
|
@ -0,0 +1,67 @@
|
|||
--- Objects/unicodeobject.c 2006/08/22 08:09:11 51465
|
||||
+++ Objects/unicodeobject.c 2006/08/22 08:25:33 51466
|
||||
@@ -1970,7 +1970,28 @@
|
||||
|
||||
static const char *hexdigit = "0123456789abcdef";
|
||||
|
||||
- repr = PyString_FromStringAndSize(NULL, 2 + 6*size + 1);
|
||||
+ /* Initial allocation is based on the longest-possible unichr
|
||||
+ escape.
|
||||
+
|
||||
+ In wide (UTF-32) builds '\U00xxxxxx' is 10 chars per source
|
||||
+ unichr, so in this case it's the longest unichr escape. In
|
||||
+ narrow (UTF-16) builds this is five chars per source unichr
|
||||
+ since there are two unichrs in the surrogate pair, so in narrow
|
||||
+ (UTF-16) builds it's not the longest unichr escape.
|
||||
+
|
||||
+ In wide or narrow builds '\uxxxx' is 6 chars per source unichr,
|
||||
+ so in the narrow (UTF-16) build case it's the longest unichr
|
||||
+ escape.
|
||||
+ */
|
||||
+
|
||||
+ repr = PyString_FromStringAndSize(NULL,
|
||||
+ 2
|
||||
+#ifdef Py_UNICODE_WIDE
|
||||
+ + 10*size
|
||||
+#else
|
||||
+ + 6*size
|
||||
+#endif
|
||||
+ + 1);
|
||||
if (repr == NULL)
|
||||
return NULL;
|
||||
|
||||
@@ -1995,15 +2016,6 @@
|
||||
#ifdef Py_UNICODE_WIDE
|
||||
/* Map 21-bit characters to '\U00xxxxxx' */
|
||||
else if (ch >= 0x10000) {
|
||||
- int offset = p - PyString_AS_STRING(repr);
|
||||
-
|
||||
- /* Resize the string if necessary */
|
||||
- if (offset + 12 > PyString_GET_SIZE(repr)) {
|
||||
- if (_PyString_Resize(&repr, PyString_GET_SIZE(repr) + 100))
|
||||
- return NULL;
|
||||
- p = PyString_AS_STRING(repr) + offset;
|
||||
- }
|
||||
-
|
||||
*p++ = '\\';
|
||||
*p++ = 'U';
|
||||
*p++ = hexdigit[(ch >> 28) & 0x0000000F];
|
||||
@@ -2016,8 +2028,8 @@
|
||||
*p++ = hexdigit[ch & 0x0000000F];
|
||||
continue;
|
||||
}
|
||||
-#endif
|
||||
- /* Map UTF-16 surrogate pairs to Unicode \UXXXXXXXX escapes */
|
||||
+#else
|
||||
+ /* Map UTF-16 surrogate pairs to '\U00xxxxxx' */
|
||||
else if (ch >= 0xD800 && ch < 0xDC00) {
|
||||
Py_UNICODE ch2;
|
||||
Py_UCS4 ucs;
|
||||
@@ -2042,6 +2054,7 @@
|
||||
s--;
|
||||
size++;
|
||||
}
|
||||
+#endif
|
||||
|
||||
/* Map 16-bit characters to '\uxxxx' */
|
||||
if (ch >= 256) {
|
Loading…
Reference in a new issue