summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAlexander Belopolsky <alexander.belopolsky@gmail.com>2010-12-23 02:27:37 (GMT)
committerAlexander Belopolsky <alexander.belopolsky@gmail.com>2010-12-23 02:27:37 (GMT)
commit86f65d5dbb2543d5b30b6f9450994bebcbc5d53e (patch)
tree56fc8827c2ec064a341d757f7cf510ebed93d26a
parent70df8f8c6779d0ecfd364b895ee6a422bc7699c5 (diff)
downloadcpython-86f65d5dbb2543d5b30b6f9450994bebcbc5d53e.zip
cpython-86f65d5dbb2543d5b30b6f9450994bebcbc5d53e.tar.gz
cpython-86f65d5dbb2543d5b30b6f9450994bebcbc5d53e.tar.bz2
Issue #10254: Fixed a crash and a regression introduced by the implementation of PRI 29.
-rw-r--r--Lib/test/test_normalization.py3
-rw-r--r--Lib/test/test_unicodedata.py15
-rw-r--r--Modules/unicodedata.c13
3 files changed, 23 insertions, 8 deletions
diff --git a/Lib/test/test_normalization.py b/Lib/test/test_normalization.py
index fa9611e..e3e2560 100644
--- a/Lib/test/test_normalization.py
+++ b/Lib/test/test_normalization.py
@@ -55,9 +55,6 @@ class NormalizationTest(unittest.TestCase):
if line.startswith("@Part"):
part = line.split()[0]
continue
- if part == "@Part3":
- # XXX we don't support PRI #29 yet, so skip these tests for now
- continue
try:
c1,c2,c3,c4,c5 = [unistr(x) for x in line.split(';')[:-1]]
except RangeError:
diff --git a/Lib/test/test_unicodedata.py b/Lib/test/test_unicodedata.py
index 1cb36fe..9744256 100644
--- a/Lib/test/test_unicodedata.py
+++ b/Lib/test/test_unicodedata.py
@@ -188,9 +188,22 @@ class UnicodeFunctionsTest(UnicodeDatabaseTest):
def test_pr29(self):
# http://www.unicode.org/review/pr-29.html
- for text in ("\u0b47\u0300\u0b3e", "\u1100\u0300\u1161"):
+ # See issues #1054943 and #10254.
+ composed = ("\u0b47\u0300\u0b3e", "\u1100\u0300\u1161",
+ 'Li\u030dt-s\u1e73\u0301',
+ '\u092e\u093e\u0930\u094d\u0915 \u091c\u093c'
+ + '\u0941\u0915\u0947\u0930\u092c\u0930\u094d\u0917',
+ '\u0915\u093f\u0930\u094d\u0917\u093f\u091c\u093c'
+ + '\u0938\u094d\u0924\u093e\u0928')
+ for text in composed:
self.assertEqual(self.db.normalize('NFC', text), text)
+ def test_issue10254(self):
+ # Crash reported in #10254
+ a = 'C\u0338' * 20 + 'C\u0327'
+ b = 'C\u0338' * 20 + '\xC7'
+ self.assertEqual(self.db.normalize('NFC', a), b)
+
def test_east_asian_width(self):
eaw = self.db.east_asian_width
self.assertRaises(TypeError, eaw, b'a')
diff --git a/Modules/unicodedata.c b/Modules/unicodedata.c
index 233f8e0..bd96e36 100644
--- a/Modules/unicodedata.c
+++ b/Modules/unicodedata.c
@@ -684,10 +684,14 @@ nfc_nfkc(PyObject *self, PyObject *input, int k)
comb = 0;
while (i1 < end) {
int comb1 = _getrecord_ex(*i1)->combining;
- if (comb && (comb1 == 0 || comb == comb1)) {
- /* Character is blocked. */
- i1++;
- continue;
+ if (comb) {
+ if (comb1 == 0)
+ break;
+ if (comb >= comb1) {
+ /* Character is blocked. */
+ i1++;
+ continue;
+ }
}
l = find_nfc_index(self, nfc_last, *i1);
/* *i1 cannot be combined with *i. If *i1
@@ -711,6 +715,7 @@ nfc_nfkc(PyObject *self, PyObject *input, int k)
/* Replace the original character. */
*i = code;
/* Mark the second character unused. */
+ assert(cskipped < 20);
skipped[cskipped++] = i1;
i1++;
f = find_nfc_index(self, nfc_first, *i);