|
59 | 59 | #include "pgstat.h"
|
60 | 60 | #include "storage/fd.h"
|
61 | 61 | #include "storage/shmem.h"
|
| 62 | +#include "storage/smgr.h" |
62 | 63 |
|
63 | 64 | #define SlruFileName(ctl, path, seg) \
|
64 | 65 | snprintf(path, MAXPGPATH, "%s/%04X", (ctl)->Dir, seg)
|
@@ -617,6 +618,66 @@ SimpleLruWritePage(SlruCtl ctl, int slotno)
|
617 | 618 | SlruInternalWritePage(ctl, slotno, NULL);
|
618 | 619 | }
|
619 | 620 |
|
| 621 | + |
| 622 | +/* |
| 623 | + * NEON: we do not want to include large pg_xact/multixact files in basebackup and prefer |
| 624 | + * to download them on demand to reduce startup time. |
| 625 | + * If SLRU segment is not found, we try to download it from page server |
| 626 | + */ |
| 627 | +static int |
| 628 | +SimpleLruDownloadSegment(SlruCtl ctl, int pageno, char const* path) |
| 629 | +{ |
| 630 | + int segno; |
| 631 | + int fd = -1; |
| 632 | + int n_blocks; |
| 633 | + char* buffer; |
| 634 | + |
| 635 | + static SMgrRelationData dummy_smgr_rel = {0}; |
| 636 | + |
| 637 | + /* If page is greater than latest written page, then do not try to download segment from server */ |
| 638 | + if (ctl->PagePrecedes(ctl->shared->latest_page_number, pageno)) |
| 639 | + return -1; |
| 640 | + |
| 641 | + if (!dummy_smgr_rel.smgr) |
| 642 | + { |
| 643 | + RelFileNode rnode = {0}; |
| 644 | + dummy_smgr_rel.smgr = smgr(InvalidBackendId, rnode); |
| 645 | + } |
| 646 | + segno = pageno / SLRU_PAGES_PER_SEGMENT; |
| 647 | + |
| 648 | + buffer = palloc(BLCKSZ * SLRU_PAGES_PER_SEGMENT); |
| 649 | + n_blocks = smgr_read_slru_segment(&dummy_smgr_rel, path, segno, buffer); |
| 650 | + if (n_blocks > 0) |
| 651 | + { |
| 652 | + fd = OpenTransientFile(path, O_RDWR | O_CREAT | PG_BINARY); |
| 653 | + if (fd < 0) |
| 654 | + { |
| 655 | + slru_errcause = SLRU_OPEN_FAILED; |
| 656 | + slru_errno = errno; |
| 657 | + pfree(buffer); |
| 658 | + return -1; |
| 659 | + } |
| 660 | + errno = 0; |
| 661 | + pgstat_report_wait_start(WAIT_EVENT_SLRU_WRITE); |
| 662 | + if (pg_pwrite(fd, buffer, n_blocks*BLCKSZ, 0) != n_blocks*BLCKSZ) |
| 663 | + { |
| 664 | + pgstat_report_wait_end(); |
| 665 | + /* if write didn't set errno, assume problem is no disk space */ |
| 666 | + if (errno == 0) |
| 667 | + errno = ENOSPC; |
| 668 | + slru_errcause = SLRU_WRITE_FAILED; |
| 669 | + slru_errno = errno; |
| 670 | + |
| 671 | + CloseTransientFile(fd); |
| 672 | + pfree(buffer); |
| 673 | + return -1; |
| 674 | + } |
| 675 | + pgstat_report_wait_end(); |
| 676 | + } |
| 677 | + pfree(buffer); |
| 678 | + return fd; |
| 679 | +} |
| 680 | + |
620 | 681 | /*
|
621 | 682 | * Return whether the given page exists on disk.
|
622 | 683 | *
|
@@ -644,12 +705,18 @@ SimpleLruDoesPhysicalPageExist(SlruCtl ctl, int pageno)
|
644 | 705 | {
|
645 | 706 | /* expected: file doesn't exist */
|
646 | 707 | if (errno == ENOENT)
|
647 |
| - return false; |
648 |
| - |
649 |
| - /* report error normally */ |
650 |
| - slru_errcause = SLRU_OPEN_FAILED; |
651 |
| - slru_errno = errno; |
652 |
| - SlruReportIOError(ctl, pageno, 0); |
| 708 | + { |
| 709 | + fd = SimpleLruDownloadSegment(ctl, pageno, path); |
| 710 | + if (fd < 0) |
| 711 | + return false; |
| 712 | + } |
| 713 | + else |
| 714 | + { |
| 715 | + /* report error normally */ |
| 716 | + slru_errcause = SLRU_OPEN_FAILED; |
| 717 | + slru_errno = errno; |
| 718 | + SlruReportIOError(ctl, pageno, 0); |
| 719 | + } |
653 | 720 | }
|
654 | 721 |
|
655 | 722 | if ((endpos = lseek(fd, 0, SEEK_END)) < 0)
|
@@ -703,18 +770,30 @@ SlruPhysicalReadPage(SlruCtl ctl, int pageno, int slotno)
|
703 | 770 | fd = OpenTransientFile(path, O_RDONLY | PG_BINARY);
|
704 | 771 | if (fd < 0)
|
705 | 772 | {
|
706 |
| - if (errno != ENOENT || !InRecovery) |
| 773 | + if (errno != ENOENT) |
707 | 774 | {
|
708 | 775 | slru_errcause = SLRU_OPEN_FAILED;
|
709 | 776 | slru_errno = errno;
|
710 | 777 | return false;
|
711 | 778 | }
|
712 |
| - |
713 |
| - ereport(LOG, |
714 |
| - (errmsg("file \"%s\" doesn't exist, reading as zeroes", |
715 |
| - path))); |
716 |
| - MemSet(shared->page_buffer[slotno], 0, BLCKSZ); |
717 |
| - return true; |
| 779 | + fd = SimpleLruDownloadSegment(ctl, pageno, path); |
| 780 | + if (fd < 0) |
| 781 | + { |
| 782 | + if (!InRecovery) |
| 783 | + { |
| 784 | + slru_errcause = SLRU_OPEN_FAILED; |
| 785 | + slru_errno = errno; |
| 786 | + return false; |
| 787 | + } |
| 788 | + else |
| 789 | + { |
| 790 | + ereport(LOG, |
| 791 | + (errmsg("file \"%s\" doesn't exist, reading as zeroes", |
| 792 | + path))); |
| 793 | + MemSet(shared->page_buffer[slotno], 0, BLCKSZ); |
| 794 | + return true; |
| 795 | + } |
| 796 | + } |
718 | 797 | }
|
719 | 798 |
|
720 | 799 | errno = 0;
|
|
0 commit comments