start_ptr = cur_ptr;
                cstate->raw_fields[fieldno] = output_ptr;
 
-               /* Scan data for field */
+               /*
+                * Scan data for field.
+                *
+                * Note that in this loop, we are scanning to locate the end of field
+                * and also speculatively performing de-escaping.  Once we find the
+                * end-of-field, we can match the raw field contents against the null
+                * marker string.  Only after that comparison fails do we know that
+                * de-escaping is actually the right thing to do; therefore we *must
+                * not* throw any syntax errors before we've done the null-marker
+                * check.
+                */
                for (;;)
                {
                        char            c;
                        *output_ptr++ = c;
                }
 
-               /* Terminate attribute value in output area */
-               *output_ptr++ = '\0';
-
-               /*
-                * If we de-escaped a non-7-bit-ASCII char, make sure we still have
-                * valid data for the db encoding. Avoid calling strlen here for the
-                * sake of efficiency.
-                */
-               if (saw_non_ascii)
-               {
-                       char       *fld = cstate->raw_fields[fieldno];
-
-                       pg_verifymbstr(fld, output_ptr - (fld + 1), false);
-               }
-
                /* Check whether raw input matched null marker */
                input_len = end_ptr - start_ptr;
                if (input_len == cstate->null_print_len &&
                        strncmp(start_ptr, cstate->null_print, input_len) == 0)
                        cstate->raw_fields[fieldno] = NULL;
+               else
+               {
+                       /*
+                        * At this point we know the field is supposed to contain data.
+                        *
+                        * If we de-escaped any non-7-bit-ASCII chars, make sure the
+                        * resulting string is valid data for the db encoding.
+                        */
+                       if (saw_non_ascii)
+                       {
+                               char       *fld = cstate->raw_fields[fieldno];
+
+                               pg_verifymbstr(fld, output_ptr - fld, false);
+                       }
+               }
+
+               /* Terminate attribute value in output area */
+               *output_ptr++ = '\0';
 
                fieldno++;
                /* Done if we hit EOL instead of a delim */
 
 \.b
 c\.d
 "\."
+-- test handling of nonstandard null marker that violates escaping rules
+CREATE TEMP TABLE testnull(a int, b text);
+INSERT INTO testnull VALUES (1, E'\\0'), (NULL, NULL);
+COPY testnull TO stdout WITH NULL AS E'\\0';
+1      \\0
+\0     \0
+COPY testnull FROM stdin WITH NULL AS E'\\0';
+SELECT * FROM testnull;
+ a  | b  
+----+----
+  1 | \0
+    | 
+ 42 | \0
+    | 
+(4 rows)
+
 DROP TABLE x, y;
 DROP FUNCTION fn_x_before();
 DROP FUNCTION fn_x_after();
 
 
 COPY testeoc TO stdout CSV;
 
+-- test handling of nonstandard null marker that violates escaping rules
+
+CREATE TEMP TABLE testnull(a int, b text);
+INSERT INTO testnull VALUES (1, E'\\0'), (NULL, NULL);
+
+COPY testnull TO stdout WITH NULL AS E'\\0';
+
+COPY testnull FROM stdin WITH NULL AS E'\\0';
+42     \\0
+\0     \0
+\.
+
+SELECT * FROM testnull;
+
+
 DROP TABLE x, y;
 DROP FUNCTION fn_x_before();
 DROP FUNCTION fn_x_after();