Line data Source code
1 : use std::{ffi::CString, sync::Arc};
2 :
3 : use byteorder::{LittleEndian, WriteBytesExt};
4 : use crc32c::crc32c_append;
5 : use parking_lot::{Mutex, MutexGuard};
6 : use postgres_ffi::{
7 : pg_constants::{
8 : RM_LOGICALMSG_ID, XLOG_LOGICAL_MESSAGE, XLP_LONG_HEADER, XLR_BLOCK_ID_DATA_LONG,
9 : XLR_BLOCK_ID_DATA_SHORT,
10 : },
11 : v16::{
12 : wal_craft_test_export::{XLogLongPageHeaderData, XLogPageHeaderData, XLOG_PAGE_MAGIC},
13 : xlog_utils::{
14 : XLogSegNoOffsetToRecPtr, XlLogicalMessage, XLOG_RECORD_CRC_OFFS,
15 : XLOG_SIZE_OF_XLOG_LONG_PHD, XLOG_SIZE_OF_XLOG_RECORD, XLOG_SIZE_OF_XLOG_SHORT_PHD,
16 : XLP_FIRST_IS_CONTRECORD,
17 : },
18 : XLogRecord,
19 : },
20 : WAL_SEGMENT_SIZE, XLOG_BLCKSZ,
21 : };
22 : use utils::lsn::Lsn;
23 :
24 : use super::block_storage::BlockStorage;
25 :
26 : /// Simulation implementation of walproposer WAL storage.
27 : pub struct DiskWalProposer {
28 : state: Mutex<State>,
29 : }
30 :
31 : impl DiskWalProposer {
32 73081 : pub fn new() -> Arc<DiskWalProposer> {
33 73081 : Arc::new(DiskWalProposer {
34 73081 : state: Mutex::new(State {
35 73081 : internal_available_lsn: Lsn(0),
36 73081 : prev_lsn: Lsn(0),
37 73081 : disk: BlockStorage::new(),
38 73081 : }),
39 73081 : })
40 73081 : }
41 :
42 76703 : pub fn lock(&self) -> MutexGuard<State> {
43 76703 : self.state.lock()
44 76703 : }
45 : }
46 :
47 : pub struct State {
48 : // flush_lsn
49 : internal_available_lsn: Lsn,
50 : // needed for WAL generation
51 : prev_lsn: Lsn,
52 : // actual WAL storage
53 : disk: BlockStorage,
54 : }
55 :
56 : impl State {
57 26978 : pub fn read(&self, pos: u64, buf: &mut [u8]) {
58 26978 : self.disk.read(pos, buf);
59 26978 : // TODO: fail on reading uninitialized data
60 26978 : }
61 :
62 136878 : pub fn write(&mut self, pos: u64, buf: &[u8]) {
63 136878 : self.disk.write(pos, buf);
64 136878 : }
65 :
66 : /// Update the internal available LSN to the given value.
67 2821 : pub fn reset_to(&mut self, lsn: Lsn) {
68 2821 : self.internal_available_lsn = lsn;
69 2821 : }
70 :
71 : /// Get current LSN.
72 11520 : pub fn flush_rec_ptr(&self) -> Lsn {
73 11520 : self.internal_available_lsn
74 11520 : }
75 :
76 : /// Generate a new WAL record at the current LSN.
77 33710 : pub fn insert_logical_message(&mut self, prefix: &str, msg: &[u8]) -> anyhow::Result<()> {
78 33710 : let prefix_cstr = CString::new(prefix)?;
79 33710 : let prefix_bytes = prefix_cstr.as_bytes_with_nul();
80 33710 :
81 33710 : let lm = XlLogicalMessage {
82 33710 : db_id: 0,
83 33710 : transactional: 0,
84 33710 : prefix_size: prefix_bytes.len() as ::std::os::raw::c_ulong,
85 33710 : message_size: msg.len() as ::std::os::raw::c_ulong,
86 33710 : };
87 33710 :
88 33710 : let record_bytes = lm.encode();
89 33710 : let rdatas: Vec<&[u8]> = vec![&record_bytes, prefix_bytes, msg];
90 33710 : insert_wal_record(self, rdatas, RM_LOGICALMSG_ID, XLOG_LOGICAL_MESSAGE)
91 33710 : }
92 : }
93 :
94 33710 : fn insert_wal_record(
95 33710 : state: &mut State,
96 33710 : rdatas: Vec<&[u8]>,
97 33710 : rmid: u8,
98 33710 : info: u8,
99 33710 : ) -> anyhow::Result<()> {
100 33710 : // bytes right after the header, in the same rdata block
101 33710 : let mut scratch = Vec::new();
102 101130 : let mainrdata_len: usize = rdatas.iter().map(|rdata| rdata.len()).sum();
103 33710 :
104 33710 : if mainrdata_len > 0 {
105 33710 : if mainrdata_len > 255 {
106 0 : scratch.push(XLR_BLOCK_ID_DATA_LONG);
107 0 : // TODO: verify endiness
108 0 : let _ = scratch.write_u32::<LittleEndian>(mainrdata_len as u32);
109 33710 : } else {
110 33710 : scratch.push(XLR_BLOCK_ID_DATA_SHORT);
111 33710 : scratch.push(mainrdata_len as u8);
112 33710 : }
113 0 : }
114 :
115 33710 : let total_len: u32 = (XLOG_SIZE_OF_XLOG_RECORD + scratch.len() + mainrdata_len) as u32;
116 33710 : let size = maxalign(total_len);
117 33710 : assert!(size as usize > XLOG_SIZE_OF_XLOG_RECORD);
118 :
119 33710 : let start_bytepos = recptr_to_bytepos(state.internal_available_lsn);
120 33710 : let end_bytepos = start_bytepos + size as u64;
121 33710 :
122 33710 : let start_recptr = bytepos_to_recptr(start_bytepos);
123 33710 : let end_recptr = bytepos_to_recptr(end_bytepos);
124 33710 :
125 33710 : assert!(recptr_to_bytepos(start_recptr) == start_bytepos);
126 33710 : assert!(recptr_to_bytepos(end_recptr) == end_bytepos);
127 :
128 33710 : let mut crc = crc32c_append(0, &scratch);
129 134840 : for rdata in &rdatas {
130 101130 : crc = crc32c_append(crc, rdata);
131 101130 : }
132 :
133 33710 : let mut header = XLogRecord {
134 33710 : xl_tot_len: total_len,
135 33710 : xl_xid: 0,
136 33710 : xl_prev: state.prev_lsn.0,
137 33710 : xl_info: info,
138 33710 : xl_rmid: rmid,
139 33710 : __bindgen_padding_0: [0u8; 2usize],
140 33710 : xl_crc: crc,
141 33710 : };
142 :
143 : // now we have the header and can finish the crc
144 33710 : let header_bytes = header.encode()?;
145 33710 : let crc = crc32c_append(crc, &header_bytes[0..XLOG_RECORD_CRC_OFFS]);
146 33710 : header.xl_crc = crc;
147 :
148 33710 : let mut header_bytes = header.encode()?.to_vec();
149 33710 : assert!(header_bytes.len() == XLOG_SIZE_OF_XLOG_RECORD);
150 :
151 33710 : header_bytes.extend_from_slice(&scratch);
152 33710 :
153 33710 : // finish rdatas
154 33710 : let mut rdatas = rdatas;
155 33710 : rdatas.insert(0, &header_bytes);
156 33710 :
157 33710 : write_walrecord_to_disk(state, total_len as u64, rdatas, start_recptr, end_recptr)?;
158 :
159 33710 : state.internal_available_lsn = end_recptr;
160 33710 : state.prev_lsn = start_recptr;
161 33710 : Ok(())
162 33710 : }
163 :
164 33710 : fn write_walrecord_to_disk(
165 33710 : state: &mut State,
166 33710 : total_len: u64,
167 33710 : rdatas: Vec<&[u8]>,
168 33710 : start: Lsn,
169 33710 : end: Lsn,
170 33710 : ) -> anyhow::Result<()> {
171 33710 : let mut curr_ptr = start;
172 33710 : let mut freespace = insert_freespace(curr_ptr);
173 33710 : let mut written: usize = 0;
174 33710 :
175 33710 : assert!(freespace >= std::mem::size_of::<u32>());
176 :
177 168550 : for mut rdata in rdatas {
178 135022 : while rdata.len() >= freespace {
179 182 : assert!(
180 182 : curr_ptr.segment_offset(WAL_SEGMENT_SIZE) >= XLOG_SIZE_OF_XLOG_SHORT_PHD
181 0 : || freespace == 0
182 : );
183 :
184 182 : state.write(curr_ptr.0, &rdata[..freespace]);
185 182 : rdata = &rdata[freespace..];
186 182 : written += freespace;
187 182 : curr_ptr = Lsn(curr_ptr.0 + freespace as u64);
188 182 :
189 182 : let mut new_page = XLogPageHeaderData {
190 182 : xlp_magic: XLOG_PAGE_MAGIC as u16,
191 182 : xlp_info: XLP_BKP_REMOVABLE,
192 182 : xlp_tli: 1,
193 182 : xlp_pageaddr: curr_ptr.0,
194 182 : xlp_rem_len: (total_len - written as u64) as u32,
195 182 : ..Default::default() // Put 0 in padding fields.
196 182 : };
197 182 : if new_page.xlp_rem_len > 0 {
198 164 : new_page.xlp_info |= XLP_FIRST_IS_CONTRECORD;
199 164 : }
200 :
201 182 : if curr_ptr.segment_offset(WAL_SEGMENT_SIZE) == 0 {
202 0 : new_page.xlp_info |= XLP_LONG_HEADER;
203 0 : let long_page = XLogLongPageHeaderData {
204 0 : std: new_page,
205 0 : xlp_sysid: 0,
206 0 : xlp_seg_size: WAL_SEGMENT_SIZE as u32,
207 0 : xlp_xlog_blcksz: XLOG_BLCKSZ as u32,
208 0 : };
209 0 : let header_bytes = long_page.encode()?;
210 0 : assert!(header_bytes.len() == XLOG_SIZE_OF_XLOG_LONG_PHD);
211 0 : state.write(curr_ptr.0, &header_bytes);
212 0 : curr_ptr = Lsn(curr_ptr.0 + header_bytes.len() as u64);
213 : } else {
214 182 : let header_bytes = new_page.encode()?;
215 182 : assert!(header_bytes.len() == XLOG_SIZE_OF_XLOG_SHORT_PHD);
216 182 : state.write(curr_ptr.0, &header_bytes);
217 182 : curr_ptr = Lsn(curr_ptr.0 + header_bytes.len() as u64);
218 : }
219 182 : freespace = insert_freespace(curr_ptr);
220 : }
221 :
222 134840 : assert!(
223 134840 : curr_ptr.segment_offset(WAL_SEGMENT_SIZE) >= XLOG_SIZE_OF_XLOG_SHORT_PHD
224 0 : || rdata.is_empty()
225 : );
226 134840 : state.write(curr_ptr.0, rdata);
227 134840 : curr_ptr = Lsn(curr_ptr.0 + rdata.len() as u64);
228 134840 : written += rdata.len();
229 134840 : freespace -= rdata.len();
230 : }
231 :
232 33710 : assert!(written == total_len as usize);
233 33710 : curr_ptr.0 = maxalign(curr_ptr.0);
234 33710 : assert!(curr_ptr == end);
235 33710 : Ok(())
236 33710 : }
237 :
238 67420 : fn maxalign<T>(size: T) -> T
239 67420 : where
240 67420 : T: std::ops::BitAnd<Output = T>
241 67420 : + std::ops::Add<Output = T>
242 67420 : + std::ops::Not<Output = T>
243 67420 : + From<u8>,
244 67420 : {
245 67420 : (size + T::from(7)) & !T::from(7)
246 67420 : }
247 :
248 33892 : fn insert_freespace(ptr: Lsn) -> usize {
249 33892 : if ptr.block_offset() == 0 {
250 0 : 0
251 : } else {
252 33892 : (XLOG_BLCKSZ as u64 - ptr.block_offset()) as usize
253 : }
254 33892 : }
255 :
256 : const XLP_BKP_REMOVABLE: u16 = 0x0004;
257 : const USABLE_BYTES_IN_PAGE: u64 = (XLOG_BLCKSZ - XLOG_SIZE_OF_XLOG_SHORT_PHD) as u64;
258 : const USABLE_BYTES_IN_SEGMENT: u64 = ((WAL_SEGMENT_SIZE / XLOG_BLCKSZ) as u64
259 : * USABLE_BYTES_IN_PAGE)
260 : - (XLOG_SIZE_OF_XLOG_RECORD - XLOG_SIZE_OF_XLOG_SHORT_PHD) as u64;
261 :
262 67420 : fn bytepos_to_recptr(bytepos: u64) -> Lsn {
263 67420 : let fullsegs = bytepos / USABLE_BYTES_IN_SEGMENT;
264 67420 : let mut bytesleft = bytepos % USABLE_BYTES_IN_SEGMENT;
265 :
266 67420 : let seg_offset = if bytesleft < (XLOG_BLCKSZ - XLOG_SIZE_OF_XLOG_SHORT_PHD) as u64 {
267 : // fits on first page of segment
268 0 : bytesleft + XLOG_SIZE_OF_XLOG_SHORT_PHD as u64
269 : } else {
270 : // account for the first page on segment with long header
271 67420 : bytesleft -= (XLOG_BLCKSZ - XLOG_SIZE_OF_XLOG_SHORT_PHD) as u64;
272 67420 : let fullpages = bytesleft / USABLE_BYTES_IN_PAGE;
273 67420 : bytesleft %= USABLE_BYTES_IN_PAGE;
274 67420 :
275 67420 : XLOG_BLCKSZ as u64
276 67420 : + fullpages * XLOG_BLCKSZ as u64
277 67420 : + bytesleft
278 67420 : + XLOG_SIZE_OF_XLOG_SHORT_PHD as u64
279 : };
280 :
281 67420 : Lsn(XLogSegNoOffsetToRecPtr(
282 67420 : fullsegs,
283 67420 : seg_offset as u32,
284 67420 : WAL_SEGMENT_SIZE,
285 67420 : ))
286 67420 : }
287 :
288 101130 : fn recptr_to_bytepos(ptr: Lsn) -> u64 {
289 101130 : let fullsegs = ptr.segment_number(WAL_SEGMENT_SIZE);
290 101130 : let offset = ptr.segment_offset(WAL_SEGMENT_SIZE) as u64;
291 101130 :
292 101130 : let fullpages = offset / XLOG_BLCKSZ as u64;
293 101130 : let offset = offset % XLOG_BLCKSZ as u64;
294 101130 :
295 101130 : if fullpages == 0 {
296 0 : fullsegs * USABLE_BYTES_IN_SEGMENT
297 0 : + if offset > 0 {
298 0 : assert!(offset >= XLOG_SIZE_OF_XLOG_SHORT_PHD as u64);
299 0 : offset - XLOG_SIZE_OF_XLOG_SHORT_PHD as u64
300 : } else {
301 0 : 0
302 : }
303 : } else {
304 101130 : fullsegs * USABLE_BYTES_IN_SEGMENT
305 101130 : + (XLOG_BLCKSZ - XLOG_SIZE_OF_XLOG_SHORT_PHD) as u64
306 101130 : + (fullpages - 1) * USABLE_BYTES_IN_PAGE
307 101130 : + if offset > 0 {
308 101130 : assert!(offset >= XLOG_SIZE_OF_XLOG_SHORT_PHD as u64);
309 101130 : offset - XLOG_SIZE_OF_XLOG_SHORT_PHD as u64
310 : } else {
311 0 : 0
312 : }
313 : }
314 101130 : }
|