]> granicus.if.org Git - postgresql/commitdiff
Fix corrupt GIN_SEGMENT_ADDITEMS WAL records on big-endian hardware.
authorTom Lane <tgl@sss.pgh.pa.us>
Sat, 3 Sep 2016 17:28:53 +0000 (13:28 -0400)
committerTom Lane <tgl@sss.pgh.pa.us>
Sat, 3 Sep 2016 17:28:53 +0000 (13:28 -0400)
computeLeafRecompressWALData() tried to produce a uint16 WAL log field by
memcpy'ing the first two bytes of an int-sized variable.  That accidentally
works on little-endian hardware, but not at all on big-endian.  Replay then
thinks it's looking at an ADDITEMS action with zero entries, and reads the
first two bytes of the first TID therein as the next segno/action,
typically leading to "unexpected GIN leaf action" errors during replay.
Even if replay failed to crash, the resulting GIN index page would surely
be incorrect.  To fix, just declare the variable as uint16 instead.

Per bug #14295 from Spencer Thomason (much thanks to Spencer for turning
his problem into a self-contained test case).  This likely also explains
a previous report of the same symptom from Bernd Helmle.

Back-patch to 9.4 where the problem was introduced (by commit 14d02f0bb).

Discussion: <20160826072658.15676.7628@wrigleys.postgresql.org>
Possible-Report: <2DA7350F7296B2A142272901@eje.land.credativ.lan>

src/backend/access/gin/gindatapage.c

index 97c8bf78e7e363337c41a6a99150a48b325a2b4b..276376a6f1de0e7c8c9f7dce59f047833ed2441c 100644 (file)
@@ -86,7 +86,7 @@ typedef struct
        char            action;
 
        ItemPointerData *modifieditems;
-       int                     nmodifieditems;
+       uint16          nmodifieditems;
 
        /*
         * The following fields represent the items in this segment. If 'items' is