Commit 1db12da8 authored by Andres Freund's avatar Andres Freund

Fix unaligned memory access in xlog parsing due to replication origin patch.

ParseCommitRecord() accessed xl_xact_origin directly. But the chunks in
the commit record's data only have 4 byte alignment, whereas
xl_xact_origin's members require 8 byte alignment on some
platforms. Update comments to make not of that and copy the record to
stack local storage before reading.

With help from Stefan Kaltenbrunner in pinning down the buildfarm and
verifying the fix.
parent 484a848a
...@@ -104,10 +104,13 @@ ParseCommitRecord(uint8 info, xl_xact_commit *xlrec, xl_xact_parsed_commit *pars ...@@ -104,10 +104,13 @@ ParseCommitRecord(uint8 info, xl_xact_commit *xlrec, xl_xact_parsed_commit *pars
if (parsed->xinfo & XACT_XINFO_HAS_ORIGIN) if (parsed->xinfo & XACT_XINFO_HAS_ORIGIN)
{ {
xl_xact_origin *xl_origin = (xl_xact_origin *) data; xl_xact_origin xl_origin;
parsed->origin_lsn = xl_origin->origin_lsn; /* we're only guaranteed 4 byte alignment, so copy onto stack */
parsed->origin_timestamp = xl_origin->origin_timestamp; memcpy(&xl_origin, data, sizeof(xl_origin));
parsed->origin_lsn = xl_origin.origin_lsn;
parsed->origin_timestamp = xl_origin.origin_timestamp;
data += sizeof(xl_xact_origin); data += sizeof(xl_xact_origin);
} }
......
...@@ -173,7 +173,8 @@ typedef struct xl_xact_assignment ...@@ -173,7 +173,8 @@ typedef struct xl_xact_assignment
* by a set XLOG_XACT_HAS_INFO bit in the xl_info field. * by a set XLOG_XACT_HAS_INFO bit in the xl_info field.
* *
* NB: All the individual data chunks should be sized to multiples of * NB: All the individual data chunks should be sized to multiples of
* sizeof(int) and only require int32 alignment. * sizeof(int) and only require int32 alignment. If they require bigger
* alignment, they need to be copied upon reading.
*/ */
/* sub-records for commit/abort */ /* sub-records for commit/abort */
...@@ -237,7 +238,7 @@ typedef struct xl_xact_commit ...@@ -237,7 +238,7 @@ typedef struct xl_xact_commit
/* xl_xact_relfilenodes follows if XINFO_HAS_RELFILENODES */ /* xl_xact_relfilenodes follows if XINFO_HAS_RELFILENODES */
/* xl_xact_invals follows if XINFO_HAS_INVALS */ /* xl_xact_invals follows if XINFO_HAS_INVALS */
/* xl_xact_twophase follows if XINFO_HAS_TWOPHASE */ /* xl_xact_twophase follows if XINFO_HAS_TWOPHASE */
/* xl_xact_origin follows if XINFO_HAS_ORIGIN */ /* xl_xact_origin follows if XINFO_HAS_ORIGIN, stored unaligned! */
} xl_xact_commit; } xl_xact_commit;
#define MinSizeOfXactCommit (offsetof(xl_xact_commit, xact_time) + sizeof(TimestampTz)) #define MinSizeOfXactCommit (offsetof(xl_xact_commit, xact_time) + sizeof(TimestampTz))
......
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