Commit 984d56b8 authored by Tom Lane's avatar Tom Lane

Fix another longstanding problem in copy_relation_data: it was blithely

assuming that a local char[] array would be aligned on at least a word
boundary.  There are architectures on which that is pretty much guaranteed to
NOT be the case ... and those arches also don't like non-aligned memory
accesses, meaning that log_newpage() would crash if it ever got invoked.
Even on Intel-ish machines there's a potential for a large performance penalty
from doing I/O to an inadequately aligned buffer.  So palloc it instead.

Backpatch to 8.0 --- 7.4 doesn't have this code.
parent 5b48e2ec
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/commands/tablecmds.c,v 1.336 2010/07/29 11:06:34 sriggs Exp $ * $PostgreSQL: pgsql/src/backend/commands/tablecmds.c,v 1.337 2010/07/29 19:23:20 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -7230,11 +7230,20 @@ static void ...@@ -7230,11 +7230,20 @@ static void
copy_relation_data(SMgrRelation src, SMgrRelation dst, copy_relation_data(SMgrRelation src, SMgrRelation dst,
ForkNumber forkNum, bool istemp) ForkNumber forkNum, bool istemp)
{ {
char *buf;
Page page;
bool use_wal; bool use_wal;
BlockNumber nblocks; BlockNumber nblocks;
BlockNumber blkno; BlockNumber blkno;
char buf[BLCKSZ];
Page page = (Page) buf; /*
* palloc the buffer so that it's MAXALIGN'd. If it were just a local
* char[] array, the compiler might align it on any byte boundary, which
* can seriously hurt transfer speed to and from the kernel; not to
* mention possibly making log_newpage's accesses to the page header fail.
*/
buf = (char *) palloc(BLCKSZ);
page = (Page) buf;
/* /*
* We need to log the copied data in WAL iff WAL archiving/streaming is * We need to log the copied data in WAL iff WAL archiving/streaming is
...@@ -7263,6 +7272,8 @@ copy_relation_data(SMgrRelation src, SMgrRelation dst, ...@@ -7263,6 +7272,8 @@ copy_relation_data(SMgrRelation src, SMgrRelation dst,
smgrextend(dst, forkNum, blkno, buf, true); smgrextend(dst, forkNum, blkno, buf, true);
} }
pfree(buf);
/* /*
* If the rel isn't temp, we must fsync it down to disk before it's safe * If the rel isn't temp, we must fsync it down to disk before it's safe
* to commit the transaction. (For a temp rel we don't care since the rel * to commit the transaction. (For a temp rel we don't care since the rel
......
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