Commit c444472a authored by Heikki Linnakangas's avatar Heikki Linnakangas

Fix backslash-escaping multibyte chars in COPY FROM.

If a multi-byte character is escaped with a backslash in TEXT mode input,
and the encoding is one of the client-only encodings where the bytes after
the first one can have an ASCII byte "embedded" in the char, we didn't
skip the character correctly. After a backslash, we only skipped the first
byte of the next character, so if it was a multi-byte character, we would
try to process its second byte as if it was a separate character. If it
was one of the characters with special meaning, like '\n', '\r', or
another '\\', that would cause trouble.

One such exmple is the byte sequence '\x5ca45c2e666f6f' in Big5 encoding.
That's supposed to be [backslash][two-byte character][.][f][o][o], but
because the second byte of the two-byte character is 0x5c, we incorrectly
treat it as another backslash. And because the next character is a dot, we
parse it as end-of-copy marker, and throw an "end-of-copy marker corrupt"
error.

Backpatch to all supported versions.

Reviewed-by: John Naylor, Kyotaro Horiguchi
Discussion: https://www.postgresql.org/message-id/a897f84f-8dca-8798-3139-07da5bb38728%40iki.fi
parent 5e7fa189
...@@ -1084,7 +1084,7 @@ CopyReadLineText(CopyFromState cstate) ...@@ -1084,7 +1084,7 @@ CopyReadLineText(CopyFromState cstate)
break; break;
} }
else if (!cstate->opts.csv_mode) else if (!cstate->opts.csv_mode)
{
/* /*
* If we are here, it means we found a backslash followed by * If we are here, it means we found a backslash followed by
* something other than a period. In non-CSV mode, anything * something other than a period. In non-CSV mode, anything
...@@ -1095,8 +1095,16 @@ CopyReadLineText(CopyFromState cstate) ...@@ -1095,8 +1095,16 @@ CopyReadLineText(CopyFromState cstate)
* backslashes are not special, so we want to process the * backslashes are not special, so we want to process the
* character after the backslash just like a normal character, * character after the backslash just like a normal character,
* so we don't increment in those cases. * so we don't increment in those cases.
*
* Set 'c' to skip whole character correctly in multi-byte
* encodings. If we don't have the whole character in the
* buffer yet, we might loop back to process it, after all,
* but that's OK because multi-byte characters cannot have any
* special meaning.
*/ */
raw_buf_ptr++; raw_buf_ptr++;
c = c2;
}
} }
/* /*
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment