File: | store/source/stordata.cxx |
Location: | line 517, column 5 |
Description: | Dereference of null pointer |
1 | /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ |
2 | /* |
3 | * This file is part of the LibreOffice project. |
4 | * |
5 | * This Source Code Form is subject to the terms of the Mozilla Public |
6 | * License, v. 2.0. If a copy of the MPL was not distributed with this |
7 | * file, You can obtain one at http://mozilla.org/MPL/2.0/. |
8 | * |
9 | * This file incorporates work covered by the following license notice: |
10 | * |
11 | * Licensed to the Apache Software Foundation (ASF) under one or more |
12 | * contributor license agreements. See the NOTICE file distributed |
13 | * with this work for additional information regarding copyright |
14 | * ownership. The ASF licenses this file to you under the Apache |
15 | * License, Version 2.0 (the "License"); you may not use this file |
16 | * except in compliance with the License. You may obtain a copy of |
17 | * the License at http://www.apache.org/licenses/LICENSE-2.0 . |
18 | */ |
19 | |
20 | |
21 | #include "stordata.hxx" |
22 | |
23 | #include "sal/types.h" |
24 | #include "osl/diagnose.h" |
25 | |
26 | #include "store/types.h" |
27 | #include "storbase.hxx" |
28 | #include "storbios.hxx" |
29 | |
30 | using namespace store; |
31 | |
32 | /*======================================================================== |
33 | * |
34 | * OStoreDataPageObject implementation. |
35 | * |
36 | *======================================================================*/ |
37 | /* |
38 | * guard. |
39 | */ |
40 | storeError OStoreDataPageObject::guard (sal_uInt32 nAddr) |
41 | { |
42 | return PageHolderObject< page >::guard (m_xPage, nAddr); |
43 | } |
44 | |
45 | /* |
46 | * verify. |
47 | */ |
48 | storeError OStoreDataPageObject::verify (sal_uInt32 nAddr) const |
49 | { |
50 | return PageHolderObject< page >::verify (m_xPage, nAddr); |
51 | } |
52 | |
53 | /*======================================================================== |
54 | * |
55 | * OStoreIndirectionPageObject implementation. |
56 | * |
57 | *======================================================================*/ |
58 | /* |
59 | * store_truncate_Impl (single indirect page). |
60 | */ |
61 | static storeError store_truncate_Impl ( |
62 | sal_uInt32 nAddr, |
63 | sal_uInt16 nSingle, |
64 | OStorePageBIOS &rBIOS) |
65 | { |
66 | if (nAddr != STORE_PAGE_NULL((sal_uInt32)(~0))) |
67 | { |
68 | // Load single indirect page. |
69 | OStoreIndirectionPageObject aSingle; |
70 | storeError eErrCode = rBIOS.loadObjectAt (aSingle, nAddr); |
71 | if (eErrCode == store_E_None) |
72 | { |
73 | // Truncate to 'nSingle' direct pages. |
74 | eErrCode = aSingle.truncate (nSingle, rBIOS); |
75 | if (eErrCode != store_E_None) |
76 | return eErrCode; |
77 | } |
78 | else |
79 | { |
80 | if (eErrCode != store_E_InvalidChecksum) |
81 | return eErrCode; |
82 | } |
83 | |
84 | // Check for complete truncation. |
85 | if (nSingle == 0) |
86 | { |
87 | // Free single indirect page. |
88 | eErrCode = rBIOS.free (nAddr); |
89 | if (eErrCode != store_E_None) |
90 | return eErrCode; |
91 | } |
92 | } |
93 | return store_E_None; |
94 | } |
95 | |
96 | /* |
97 | * store_truncate_Impl (double indirect page). |
98 | */ |
99 | static storeError store_truncate_Impl ( |
100 | sal_uInt32 nAddr, |
101 | sal_uInt16 nDouble, |
102 | sal_uInt16 nSingle, |
103 | OStorePageBIOS &rBIOS) |
104 | { |
105 | if (nAddr != STORE_PAGE_NULL((sal_uInt32)(~0))) |
106 | { |
107 | // Load double indirect page. |
108 | OStoreIndirectionPageObject aDouble; |
109 | storeError eErrCode = rBIOS.loadObjectAt (aDouble, nAddr); |
110 | if (eErrCode == store_E_None) |
111 | { |
112 | // Truncate to 'nDouble', 'nSingle' pages. |
113 | eErrCode = aDouble.truncate (nDouble, nSingle, rBIOS); |
114 | if (eErrCode != store_E_None) |
115 | return eErrCode; |
116 | } |
117 | else |
118 | { |
119 | if (eErrCode != store_E_InvalidChecksum) |
120 | return eErrCode; |
121 | } |
122 | |
123 | // Check for complete truncation. |
124 | if ((nDouble + nSingle) == 0) |
125 | { |
126 | // Free double indirect page. |
127 | eErrCode = rBIOS.free (nAddr); |
128 | if (eErrCode != store_E_None) |
129 | return eErrCode; |
130 | } |
131 | } |
132 | return store_E_None; |
133 | } |
134 | |
135 | /* |
136 | * store_truncate_Impl (triple indirect page). |
137 | */ |
138 | static storeError store_truncate_Impl ( |
139 | sal_uInt32 nAddr, |
140 | sal_uInt16 nTriple, |
141 | sal_uInt16 nDouble, |
142 | sal_uInt16 nSingle, |
143 | OStorePageBIOS &rBIOS) |
144 | { |
145 | if (nAddr != STORE_PAGE_NULL((sal_uInt32)(~0))) |
146 | { |
147 | // Load triple indirect page. |
148 | OStoreIndirectionPageObject aTriple; |
149 | storeError eErrCode = rBIOS.loadObjectAt (aTriple, nAddr); |
150 | if (eErrCode != store_E_None) |
151 | return eErrCode; |
152 | |
153 | // Truncate to 'nTriple', 'nDouble', 'nSingle' pages. |
154 | eErrCode = aTriple.truncate (nTriple, nDouble, nSingle, rBIOS); |
155 | if (eErrCode != store_E_None) |
156 | return eErrCode; |
157 | |
158 | // Check for complete truncation. |
159 | if ((nTriple + nDouble + nSingle) == 0) |
160 | { |
161 | // Free triple indirect page. |
162 | eErrCode = rBIOS.free (nAddr); |
163 | if (eErrCode != store_E_None) |
164 | return eErrCode; |
165 | } |
166 | } |
167 | return store_E_None; |
168 | } |
169 | |
170 | /* |
171 | * loadOrCreate. |
172 | */ |
173 | storeError OStoreIndirectionPageObject::loadOrCreate ( |
174 | sal_uInt32 nAddr, |
175 | OStorePageBIOS & rBIOS) |
176 | { |
177 | if (nAddr == STORE_PAGE_NULL((sal_uInt32)(~0))) |
178 | { |
179 | storeError eErrCode = construct<page>(rBIOS.allocator()); |
180 | if (eErrCode != store_E_None) |
181 | return eErrCode; |
182 | |
183 | eErrCode = rBIOS.allocate (*this); |
184 | if (eErrCode != store_E_None) |
185 | return eErrCode; |
186 | |
187 | // Save location pending at caller. |
188 | return store_E_Pending; |
189 | } |
190 | return rBIOS.loadObjectAt (*this, nAddr); |
191 | } |
192 | |
193 | /* |
194 | * guard. |
195 | */ |
196 | storeError OStoreIndirectionPageObject::guard (sal_uInt32 nAddr) |
197 | { |
198 | return PageHolderObject< page >::guard (m_xPage, nAddr); |
199 | } |
200 | |
201 | /* |
202 | * verify. |
203 | */ |
204 | storeError OStoreIndirectionPageObject::verify (sal_uInt32 nAddr) const |
205 | { |
206 | return PageHolderObject< page >::verify (m_xPage, nAddr); |
207 | } |
208 | |
209 | /* |
210 | * read (single indirect). |
211 | */ |
212 | storeError OStoreIndirectionPageObject::read ( |
213 | sal_uInt16 nSingle, |
214 | OStoreDataPageObject &rData, |
215 | OStorePageBIOS &rBIOS) |
216 | { |
217 | PageHolderObject< page > xImpl (m_xPage); |
218 | page const & rPage = (*xImpl); |
219 | |
220 | // Check arguments. |
221 | sal_uInt16 const nLimit = rPage.capacityCount(); |
222 | if (!(nSingle < nLimit)) |
223 | return store_E_InvalidAccess; |
224 | |
225 | // Obtain data page location. |
226 | sal_uInt32 const nAddr = store::ntohl(rPage.m_pData[nSingle]); |
227 | if (nAddr == STORE_PAGE_NULL((sal_uInt32)(~0))) |
228 | return store_E_NotExists; |
229 | |
230 | // Load data page and leave. |
231 | return rBIOS.loadObjectAt (rData, nAddr); |
232 | } |
233 | |
234 | /* |
235 | * read (double indirect). |
236 | */ |
237 | storeError OStoreIndirectionPageObject::read ( |
238 | sal_uInt16 nDouble, |
239 | sal_uInt16 nSingle, |
240 | OStoreDataPageObject &rData, |
241 | OStorePageBIOS &rBIOS) |
242 | { |
243 | PageHolderObject< page > xImpl (m_xPage); |
244 | page const & rPage = (*xImpl); |
245 | |
246 | // Check arguments. |
247 | sal_uInt16 const nLimit = rPage.capacityCount(); |
248 | if (!((nDouble < nLimit) && (nSingle < nLimit))) |
249 | return store_E_InvalidAccess; |
250 | |
251 | // Check single indirect page location. |
252 | sal_uInt32 const nAddr = store::ntohl(rPage.m_pData[nDouble]); |
253 | if (nAddr == STORE_PAGE_NULL((sal_uInt32)(~0))) |
254 | return store_E_NotExists; |
255 | |
256 | // Load single indirect page. |
257 | OStoreIndirectionPageObject aSingle; |
258 | storeError eErrCode = rBIOS.loadObjectAt (aSingle, nAddr); |
259 | if (eErrCode != store_E_None) |
260 | return eErrCode; |
261 | |
262 | // Read single indirect and leave. |
263 | return aSingle.read (nSingle, rData, rBIOS); |
264 | } |
265 | |
266 | /* |
267 | * read (triple indirect). |
268 | */ |
269 | storeError OStoreIndirectionPageObject::read ( |
270 | sal_uInt16 nTriple, |
271 | sal_uInt16 nDouble, |
272 | sal_uInt16 nSingle, |
273 | OStoreDataPageObject &rData, |
274 | OStorePageBIOS &rBIOS) |
275 | { |
276 | PageHolderObject< page > xImpl (m_xPage); |
277 | page const & rPage = (*xImpl); |
278 | |
279 | // Check arguments. |
280 | sal_uInt16 const nLimit = rPage.capacityCount(); |
281 | if (!((nTriple < nLimit) && (nDouble < nLimit) && (nSingle < nLimit))) |
282 | return store_E_InvalidAccess; |
283 | |
284 | // Check double indirect page location. |
285 | sal_uInt32 const nAddr = store::ntohl(rPage.m_pData[nTriple]); |
286 | if (nAddr == STORE_PAGE_NULL((sal_uInt32)(~0))) |
287 | return store_E_NotExists; |
288 | |
289 | // Load double indirect page. |
290 | OStoreIndirectionPageObject aDouble; |
291 | storeError eErrCode = rBIOS.loadObjectAt (aDouble, nAddr); |
292 | if (eErrCode != store_E_None) |
293 | return eErrCode; |
294 | |
295 | // Read double indirect and leave. |
296 | return aDouble.read (nDouble, nSingle, rData, rBIOS); |
297 | } |
298 | |
299 | /* |
300 | * write (single indirect). |
301 | */ |
302 | storeError OStoreIndirectionPageObject::write ( |
303 | sal_uInt16 nSingle, |
304 | OStoreDataPageObject &rData, |
305 | OStorePageBIOS &rBIOS) |
306 | { |
307 | PageHolderObject< page > xImpl (m_xPage); |
308 | page & rPage = (*xImpl); |
309 | |
310 | // Check arguments. |
311 | sal_uInt16 const nLimit = rPage.capacityCount(); |
312 | if (!(nSingle < nLimit)) |
313 | return store_E_InvalidAccess; |
314 | |
315 | // Obtain data page location. |
316 | sal_uInt32 const nAddr = store::ntohl(rPage.m_pData[nSingle]); |
317 | if (nAddr == STORE_PAGE_NULL((sal_uInt32)(~0))) |
318 | { |
319 | // Allocate data page. |
320 | storeError eErrCode = rBIOS.allocate (rData); |
321 | if (eErrCode != store_E_None) |
322 | return eErrCode; |
323 | |
324 | // Store data page location. |
325 | rPage.m_pData[nSingle] = store::htonl(rData.location()); |
326 | |
327 | // Save this page. |
328 | return rBIOS.saveObjectAt (*this, location()); |
329 | } |
330 | else |
331 | { |
332 | // Save data page. |
333 | return rBIOS.saveObjectAt (rData, nAddr); |
334 | } |
335 | } |
336 | |
337 | /* |
338 | * write (double indirect). |
339 | */ |
340 | storeError OStoreIndirectionPageObject::write ( |
341 | sal_uInt16 nDouble, |
342 | sal_uInt16 nSingle, |
343 | OStoreDataPageObject &rData, |
344 | OStorePageBIOS &rBIOS) |
345 | { |
346 | PageHolderObject< page > xImpl (m_xPage); |
347 | page & rPage = (*xImpl); |
348 | |
349 | // Check arguments. |
350 | sal_uInt16 const nLimit = rPage.capacityCount(); |
351 | if (!((nDouble < nLimit) && (nSingle < nLimit))) |
352 | return store_E_InvalidAccess; |
353 | |
354 | // Load or create single indirect page. |
355 | OStoreIndirectionPageObject aSingle; |
356 | storeError eErrCode = aSingle.loadOrCreate (store::ntohl(rPage.m_pData[nDouble]), rBIOS); |
357 | if (eErrCode != store_E_None) |
358 | { |
359 | if (eErrCode != store_E_Pending) |
360 | return eErrCode; |
361 | rPage.m_pData[nDouble] = store::htonl(aSingle.location()); |
362 | |
363 | eErrCode = rBIOS.saveObjectAt (*this, location()); |
364 | if (eErrCode != store_E_None) |
365 | return eErrCode; |
366 | } |
367 | |
368 | // Write single indirect and leave. |
369 | return aSingle.write (nSingle, rData, rBIOS); |
370 | } |
371 | |
372 | /* |
373 | * write (triple indirect). |
374 | */ |
375 | storeError OStoreIndirectionPageObject::write ( |
376 | sal_uInt16 nTriple, |
377 | sal_uInt16 nDouble, |
378 | sal_uInt16 nSingle, |
379 | OStoreDataPageObject &rData, |
380 | OStorePageBIOS &rBIOS) |
381 | { |
382 | PageHolderObject< page > xImpl (m_xPage); |
383 | page & rPage = (*xImpl); |
384 | |
385 | // Check arguments. |
386 | sal_uInt16 const nLimit = rPage.capacityCount(); |
387 | if (!((nTriple < nLimit) && (nDouble < nLimit) && (nSingle < nLimit))) |
388 | return store_E_InvalidAccess; |
389 | |
390 | // Load or create double indirect page. |
391 | OStoreIndirectionPageObject aDouble; |
392 | storeError eErrCode = aDouble.loadOrCreate (store::ntohl(rPage.m_pData[nTriple]), rBIOS); |
393 | if (eErrCode != store_E_None) |
394 | { |
395 | if (eErrCode != store_E_Pending) |
396 | return eErrCode; |
397 | rPage.m_pData[nTriple] = store::htonl(aDouble.location()); |
398 | |
399 | eErrCode = rBIOS.saveObjectAt (*this, location()); |
400 | if (eErrCode != store_E_None) |
401 | return eErrCode; |
402 | } |
403 | |
404 | // Write double indirect and leave. |
405 | return aDouble.write (nDouble, nSingle, rData, rBIOS); |
406 | } |
407 | |
408 | /* |
409 | * truncate (single indirect). |
410 | */ |
411 | storeError OStoreIndirectionPageObject::truncate ( |
412 | sal_uInt16 nSingle, |
413 | OStorePageBIOS & rBIOS) |
414 | { |
415 | PageHolderObject< page > xImpl (m_xPage); |
416 | page & rPage = (*xImpl); |
417 | |
418 | // Check arguments. |
419 | sal_uInt16 const nLimit = rPage.capacityCount(); |
420 | if (!(nSingle < nLimit)) |
421 | return store_E_InvalidAccess; |
422 | |
423 | // Truncate. |
424 | storeError eErrCode = store_E_None; |
425 | for (sal_uInt16 i = nLimit; i > nSingle; i--) |
426 | { |
427 | // Obtain data page location. |
428 | sal_uInt32 const nAddr = store::ntohl(rPage.m_pData[i - 1]); |
429 | if (nAddr != STORE_PAGE_NULL((sal_uInt32)(~0))) |
430 | { |
431 | // Free data page. |
432 | eErrCode = rBIOS.free (nAddr); |
433 | if (eErrCode != store_E_None) |
434 | return eErrCode; |
435 | |
436 | // Clear pointer to data page. |
437 | rPage.m_pData[i - 1] = STORE_PAGE_NULL((sal_uInt32)(~0)); |
438 | touch(); |
439 | } |
440 | } |
441 | |
442 | // Check for modified page. |
443 | if (dirty()) |
444 | { |
445 | // Save this page. |
446 | eErrCode = rBIOS.saveObjectAt (*this, location()); |
447 | } |
448 | |
449 | // Done. |
450 | return eErrCode; |
451 | } |
452 | |
453 | /* |
454 | * truncate (double indirect). |
455 | */ |
456 | storeError OStoreIndirectionPageObject::truncate ( |
457 | sal_uInt16 nDouble, |
458 | sal_uInt16 nSingle, |
459 | OStorePageBIOS &rBIOS) |
460 | { |
461 | PageHolderObject< page > xImpl (m_xPage); |
462 | page & rPage = (*xImpl); |
463 | |
464 | // Check arguments. |
465 | sal_uInt16 const nLimit = rPage.capacityCount(); |
466 | if (!((nDouble < nLimit) && (nSingle < nLimit))) |
467 | return store_E_InvalidAccess; |
468 | |
469 | // Truncate. |
470 | storeError eErrCode = store_E_None; |
471 | for (sal_uInt16 i = nLimit; i > nDouble + 1; i--) |
472 | { |
473 | // Truncate single indirect page to zero direct pages. |
474 | eErrCode = store_truncate_Impl (store::ntohl(rPage.m_pData[i - 1]), 0, rBIOS); |
475 | if (eErrCode != store_E_None) |
476 | return eErrCode; |
477 | |
478 | // Clear pointer to single indirect page. |
479 | rPage.m_pData[i - 1] = STORE_PAGE_NULL((sal_uInt32)(~0)); |
480 | touch(); |
481 | } |
482 | |
483 | // Truncate last single indirect page to 'nSingle' direct pages. |
484 | eErrCode = store_truncate_Impl (store::ntohl(rPage.m_pData[nDouble]), nSingle, rBIOS); |
485 | if (eErrCode != store_E_None) |
486 | return eErrCode; |
487 | |
488 | // Check for complete truncation. |
489 | if (nSingle == 0) |
490 | { |
491 | // Clear pointer to last single indirect page. |
492 | rPage.m_pData[nDouble] = STORE_PAGE_NULL((sal_uInt32)(~0)); |
493 | touch(); |
494 | } |
495 | |
496 | // Check for modified page. |
497 | if (dirty()) |
498 | { |
499 | // Save this page. |
500 | eErrCode = rBIOS.saveObjectAt (*this, location()); |
501 | } |
502 | |
503 | // Done. |
504 | return eErrCode; |
505 | } |
506 | |
507 | /* |
508 | * truncate (triple indirect). |
509 | */ |
510 | storeError OStoreIndirectionPageObject::truncate ( |
511 | sal_uInt16 nTriple, |
512 | sal_uInt16 nDouble, |
513 | sal_uInt16 nSingle, |
514 | OStorePageBIOS &rBIOS) |
515 | { |
516 | PageHolderObject< page > xImpl (m_xPage); |
517 | page & rPage = (*xImpl); |
Dereference of null pointer | |
518 | |
519 | // Check arguments. |
520 | sal_uInt16 const nLimit = rPage.capacityCount(); |
521 | if (!((nTriple < nLimit) && (nDouble < nLimit) && (nSingle < nLimit))) |
522 | return store_E_InvalidAccess; |
523 | |
524 | // Truncate. |
525 | storeError eErrCode = store_E_None; |
526 | for (sal_uInt16 i = nLimit; i > nTriple + 1; i--) |
527 | { |
528 | // Truncate double indirect page to zero single indirect pages. |
529 | eErrCode = store_truncate_Impl (store::ntohl(rPage.m_pData[i - 1]), 0, 0, rBIOS); |
530 | if (eErrCode != store_E_None) |
531 | return eErrCode; |
532 | |
533 | // Clear pointer to double indirect page. |
534 | rPage.m_pData[i - 1] = STORE_PAGE_NULL((sal_uInt32)(~0)); |
535 | touch(); |
536 | } |
537 | |
538 | // Truncate last double indirect page to 'nDouble', 'nSingle' pages. |
539 | eErrCode = store_truncate_Impl (store::ntohl(rPage.m_pData[nTriple]), nDouble, nSingle, rBIOS); |
540 | if (eErrCode != store_E_None) |
541 | return eErrCode; |
542 | |
543 | // Check for complete truncation. |
544 | if ((nDouble + nSingle) == 0) |
545 | { |
546 | // Clear pointer to last double indirect page. |
547 | rPage.m_pData[nTriple] = STORE_PAGE_NULL((sal_uInt32)(~0)); |
548 | touch(); |
549 | } |
550 | |
551 | // Check for modified page. |
552 | if (dirty()) |
553 | { |
554 | // Save this page. |
555 | eErrCode = rBIOS.saveObjectAt (*this, location()); |
556 | } |
557 | |
558 | // Done. |
559 | return eErrCode; |
560 | } |
561 | |
562 | /*======================================================================== |
563 | * |
564 | * OStoreDirectoryPageObject implementation. |
565 | * |
566 | *======================================================================*/ |
567 | /* |
568 | * guard. |
569 | */ |
570 | storeError OStoreDirectoryPageObject::guard (sal_uInt32 nAddr) |
571 | { |
572 | return PageHolderObject< page >::guard (m_xPage, nAddr); |
573 | } |
574 | |
575 | /* |
576 | * verify. |
577 | */ |
578 | storeError OStoreDirectoryPageObject::verify (sal_uInt32 nAddr) const |
579 | { |
580 | return PageHolderObject< page >::verify (m_xPage, nAddr); |
581 | // OLD: m_rPage.verifyVersion (STORE_MAGIC_DIRECTORYPAGE); |
582 | } |
583 | |
584 | /* |
585 | * scope (external data page; private). |
586 | */ |
587 | OStoreDirectoryPageData::ChunkScope |
588 | OStoreDirectoryPageObject::scope ( |
589 | sal_uInt32 nPage, |
590 | page::DataBlock::LinkDescriptor &rDescr) const |
591 | { |
592 | page const & rPage = PAGE(); |
593 | OStoreDirectoryDataBlock const & rDataBlock = rPage.m_aDataBlock; |
594 | |
595 | sal_uInt32 index0, index1, index2, index3; |
596 | |
597 | // direct. |
598 | sal_uInt32 nCount = rDataBlock.directCount(); |
599 | sal_uInt32 nLimit = nCount; |
600 | if (nPage < nLimit) |
601 | { |
602 | // Page to index reduction. |
603 | index0 = nPage; |
604 | |
605 | // Setup LinkDescriptor indices. |
606 | rDescr.m_nIndex0 = (sal_uInt16)(index0 & 0xffff); |
607 | |
608 | // Done. |
609 | return page::SCOPE_DIRECT; |
610 | } |
611 | nPage -= nLimit; |
612 | |
613 | // single indirect. |
614 | sal_uInt32 const nCapacity = indirect::capacityCount(rPage.m_aDescr); |
615 | nCount = rDataBlock.singleCount(); |
616 | nLimit = nCount * nCapacity; |
617 | if (nPage < nLimit) |
618 | { |
619 | // Page to index reduction. |
620 | sal_uInt32 n = nPage; |
621 | |
622 | // Reduce to single indirect i(1), direct n = i(0). |
623 | index1 = n / nCapacity; |
624 | index0 = n % nCapacity; |
625 | |
626 | // Verify reduction. |
627 | n = index1 * nCapacity + index0; |
628 | OSL_POSTCOND(n == nPage, "wrong math on indirect indices")do { if (true && (!(n == nPage))) { sal_detail_logFormat ((SAL_DETAIL_LOG_LEVEL_WARN), ("legacy.osl"), ("/usr/local/src/libreoffice/store/source/stordata.cxx" ":" "628" ": "), "%s", "wrong math on indirect indices"); } } while (false); |
629 | if (n != nPage) |
630 | return page::SCOPE_UNKNOWN; |
631 | |
632 | // Setup LinkDescriptor indices. |
633 | rDescr.m_nIndex0 = (sal_uInt16)(index0 & 0xffff); |
634 | rDescr.m_nIndex1 = (sal_uInt16)(index1 & 0xffff); |
635 | |
636 | // Done. |
637 | return page::SCOPE_SINGLE; |
638 | } |
639 | nPage -= nLimit; |
640 | |
641 | // double indirect. |
642 | nCount = rDataBlock.doubleCount(); |
643 | nLimit = nCount * nCapacity * nCapacity; |
644 | if (nPage < nLimit) |
645 | { |
646 | // Page to index reduction. |
647 | sal_uInt32 n = nPage; |
648 | |
649 | // Reduce to double indirect i(2), single indirect n = i(0). |
650 | index2 = n / (nCapacity * nCapacity); |
651 | n = n % (nCapacity * nCapacity); |
652 | |
653 | // Reduce to single indirect i(1), direct n = i(0). |
654 | index1 = n / nCapacity; |
655 | index0 = n % nCapacity; |
656 | |
657 | // Verify reduction. |
658 | n = index2 * nCapacity * nCapacity + |
659 | index1 * nCapacity + index0; |
660 | OSL_POSTCOND(n == nPage, "wrong math on double indirect indices")do { if (true && (!(n == nPage))) { sal_detail_logFormat ((SAL_DETAIL_LOG_LEVEL_WARN), ("legacy.osl"), ("/usr/local/src/libreoffice/store/source/stordata.cxx" ":" "660" ": "), "%s", "wrong math on double indirect indices" ); } } while (false); |
661 | if (n != nPage) |
662 | return page::SCOPE_UNKNOWN; |
663 | |
664 | // Setup LinkDescriptor indices. |
665 | rDescr.m_nIndex0 = (sal_uInt16)(index0 & 0xffff); |
666 | rDescr.m_nIndex1 = (sal_uInt16)(index1 & 0xffff); |
667 | rDescr.m_nIndex2 = (sal_uInt16)(index2 & 0xffff); |
668 | |
669 | // Done. |
670 | return page::SCOPE_DOUBLE; |
671 | } |
672 | nPage -= nLimit; |
673 | |
674 | // triple indirect. |
675 | nCount = rDataBlock.tripleCount(); |
676 | nLimit = nCount * nCapacity * nCapacity * nCapacity; |
677 | if (nPage < nLimit) |
678 | { |
679 | // Page to index reduction. |
680 | sal_uInt32 n = nPage; |
681 | |
682 | // Reduce to triple indirect i(3), double indirect n. |
683 | index3 = n / (nCapacity * nCapacity * nCapacity); |
684 | n = n % (nCapacity * nCapacity * nCapacity); |
685 | |
686 | // Reduce to double indirect i(2), single indirect n. |
687 | index2 = n / (nCapacity * nCapacity); |
688 | n = n % (nCapacity * nCapacity); |
689 | |
690 | // Reduce to single indirect i(1), direct n = i(0). |
691 | index1 = n / nCapacity; |
692 | index0 = n % nCapacity; |
693 | |
694 | // Verify reduction. |
695 | n = index3 * nCapacity * nCapacity * nCapacity + |
696 | index2 * nCapacity * nCapacity + |
697 | index1 * nCapacity + index0; |
698 | OSL_POSTCOND(n == nPage, "wrong math on triple indirect indices")do { if (true && (!(n == nPage))) { sal_detail_logFormat ((SAL_DETAIL_LOG_LEVEL_WARN), ("legacy.osl"), ("/usr/local/src/libreoffice/store/source/stordata.cxx" ":" "698" ": "), "%s", "wrong math on triple indirect indices" ); } } while (false); |
699 | if (n != nPage) |
700 | return page::SCOPE_UNKNOWN; |
701 | |
702 | // Setup LinkDescriptor indices. |
703 | rDescr.m_nIndex0 = (sal_uInt16)(index0 & 0xffff); |
704 | rDescr.m_nIndex1 = (sal_uInt16)(index1 & 0xffff); |
705 | rDescr.m_nIndex2 = (sal_uInt16)(index2 & 0xffff); |
706 | rDescr.m_nIndex3 = (sal_uInt16)(index3 & 0xffff); |
707 | |
708 | // Done. |
709 | return page::SCOPE_TRIPLE; |
710 | } |
711 | |
712 | // Unreachable (more than triple indirect). |
713 | return page::SCOPE_UNREACHABLE; |
714 | } |
715 | |
716 | /* |
717 | * read (external data page). |
718 | */ |
719 | storeError OStoreDirectoryPageObject::read ( |
720 | sal_uInt32 nPage, |
721 | OStoreDataPageObject &rData, |
722 | OStorePageBIOS &rBIOS) |
723 | { |
724 | // Determine scope and link indices. |
725 | page::DataBlock::LinkDescriptor aLink; |
726 | page::ChunkScope eScope = scope (nPage, aLink); |
727 | |
728 | storeError eErrCode = store_E_None; |
729 | if (eScope == page::SCOPE_DIRECT) |
730 | { |
731 | sal_uInt32 const nAddr = directLink (aLink.m_nIndex0); |
732 | if (nAddr == STORE_PAGE_NULL((sal_uInt32)(~0))) |
733 | return store_E_NotExists; |
734 | |
735 | eErrCode = rBIOS.loadObjectAt (rData, nAddr); |
736 | } |
737 | else if (eScope == page::SCOPE_SINGLE) |
738 | { |
739 | sal_uInt32 const nAddr = singleLink (aLink.m_nIndex1); |
740 | if (nAddr == STORE_PAGE_NULL((sal_uInt32)(~0))) |
741 | return store_E_NotExists; |
742 | |
743 | OStoreIndirectionPageObject aSingle; |
744 | eErrCode = rBIOS.loadObjectAt (aSingle, nAddr); |
745 | if (eErrCode != store_E_None) |
746 | return eErrCode; |
747 | |
748 | eErrCode = aSingle.read (aLink.m_nIndex0, rData, rBIOS); |
749 | } |
750 | else if (eScope == page::SCOPE_DOUBLE) |
751 | { |
752 | sal_uInt32 const nAddr = doubleLink (aLink.m_nIndex2); |
753 | if (nAddr == STORE_PAGE_NULL((sal_uInt32)(~0))) |
754 | return store_E_NotExists; |
755 | |
756 | OStoreIndirectionPageObject aDouble; |
757 | eErrCode = rBIOS.loadObjectAt (aDouble, nAddr); |
758 | if (eErrCode != store_E_None) |
759 | return eErrCode; |
760 | |
761 | eErrCode = aDouble.read (aLink.m_nIndex1, aLink.m_nIndex0, rData, rBIOS); |
762 | } |
763 | else if (eScope == page::SCOPE_TRIPLE) |
764 | { |
765 | sal_uInt32 const nAddr = tripleLink (aLink.m_nIndex3); |
766 | if (nAddr == STORE_PAGE_NULL((sal_uInt32)(~0))) |
767 | return store_E_NotExists; |
768 | |
769 | OStoreIndirectionPageObject aTriple; |
770 | eErrCode = rBIOS.loadObjectAt (aTriple, nAddr); |
771 | if (eErrCode != store_E_None) |
772 | return eErrCode; |
773 | |
774 | eErrCode = aTriple.read (aLink.m_nIndex2, aLink.m_nIndex1, aLink.m_nIndex0, rData, rBIOS); |
775 | } |
776 | else if (eScope == page::SCOPE_UNREACHABLE) |
777 | { |
778 | // Out of scope. |
779 | eErrCode = store_E_CantSeek; |
780 | } |
781 | else |
782 | { |
783 | // Unknown scope. |
784 | OSL_TRACE("OStoreDirectoryPageObject::get(): scope failed")do { if (true && (1 > 0)) { sal_detail_logFormat(( SAL_DETAIL_LOG_LEVEL_INFO), ("legacy.osl"), ("/usr/local/src/libreoffice/store/source/stordata.cxx" ":" "784" ": "), "OStoreDirectoryPageObject::get(): scope failed" ); } } while (false); |
785 | eErrCode = store_E_Unknown; |
786 | } |
787 | |
788 | // Leave. |
789 | return eErrCode; |
790 | } |
791 | |
792 | /* |
793 | * write (external data page). |
794 | */ |
795 | storeError OStoreDirectoryPageObject::write ( |
796 | sal_uInt32 nPage, |
797 | OStoreDataPageObject &rData, |
798 | OStorePageBIOS &rBIOS) |
799 | { |
800 | // Determine scope and link indices. |
801 | page::DataBlock::LinkDescriptor aLink; |
802 | page::ChunkScope eScope = scope (nPage, aLink); |
803 | |
804 | storeError eErrCode = store_E_None; |
805 | if (eScope == page::SCOPE_DIRECT) |
806 | { |
807 | sal_uInt32 const nAddr = directLink (aLink.m_nIndex0); |
808 | if (nAddr == STORE_PAGE_NULL((sal_uInt32)(~0))) |
809 | { |
810 | // Allocate data page. |
811 | eErrCode = rBIOS.allocate (rData); |
812 | if (eErrCode != store_E_None) |
813 | return eErrCode; |
814 | |
815 | // Store data page location. |
816 | directLink (aLink.m_nIndex0, rData.location()); |
817 | } |
818 | else |
819 | { |
820 | // Save data page. |
821 | eErrCode = rBIOS.saveObjectAt (rData, nAddr); |
822 | } |
823 | } |
824 | else if (eScope == page::SCOPE_SINGLE) |
825 | { |
826 | OStoreIndirectionPageObject aSingle; |
827 | eErrCode = aSingle.loadOrCreate (singleLink (aLink.m_nIndex1), rBIOS); |
828 | if (eErrCode != store_E_None) |
829 | { |
830 | if (eErrCode != store_E_Pending) |
831 | return eErrCode; |
832 | singleLink (aLink.m_nIndex1, aSingle.location()); |
833 | } |
834 | |
835 | eErrCode = aSingle.write (aLink.m_nIndex0, rData, rBIOS); |
836 | } |
837 | else if (eScope == page::SCOPE_DOUBLE) |
838 | { |
839 | OStoreIndirectionPageObject aDouble; |
840 | eErrCode = aDouble.loadOrCreate (doubleLink (aLink.m_nIndex2), rBIOS); |
841 | if (eErrCode != store_E_None) |
842 | { |
843 | if (eErrCode != store_E_Pending) |
844 | return eErrCode; |
845 | doubleLink (aLink.m_nIndex2, aDouble.location()); |
846 | } |
847 | |
848 | eErrCode = aDouble.write (aLink.m_nIndex1, aLink.m_nIndex0, rData, rBIOS); |
849 | } |
850 | else if (eScope == page::SCOPE_TRIPLE) |
851 | { |
852 | OStoreIndirectionPageObject aTriple; |
853 | eErrCode = aTriple.loadOrCreate (tripleLink (aLink.m_nIndex3), rBIOS); |
854 | if (eErrCode != store_E_None) |
855 | { |
856 | if (eErrCode != store_E_Pending) |
857 | return eErrCode; |
858 | tripleLink (aLink.m_nIndex3, aTriple.location()); |
859 | } |
860 | |
861 | eErrCode = aTriple.write (aLink.m_nIndex2, aLink.m_nIndex1, aLink.m_nIndex0, rData, rBIOS); |
862 | } |
863 | else if (eScope == page::SCOPE_UNREACHABLE) |
864 | { |
865 | // Out of scope. |
866 | eErrCode = store_E_CantSeek; |
867 | } |
868 | else |
869 | { |
870 | // Unknown scope. |
871 | OSL_TRACE("OStoreDirectoryPageObject::put(): scope failed")do { if (true && (1 > 0)) { sal_detail_logFormat(( SAL_DETAIL_LOG_LEVEL_INFO), ("legacy.osl"), ("/usr/local/src/libreoffice/store/source/stordata.cxx" ":" "871" ": "), "OStoreDirectoryPageObject::put(): scope failed" ); } } while (false); |
872 | eErrCode = store_E_Unknown; |
873 | } |
874 | |
875 | // Leave. |
876 | return eErrCode; |
877 | } |
878 | |
879 | /* |
880 | * truncate (external data page). |
881 | */ |
882 | storeError OStoreDirectoryPageObject::truncate ( |
883 | sal_uInt32 nPage, |
884 | OStorePageBIOS &rBIOS) |
885 | { |
886 | // Determine scope and link indices. |
887 | page::DataBlock::LinkDescriptor aLink; |
888 | page::ChunkScope eScope = scope (nPage, aLink); |
889 | |
890 | storeError eErrCode = store_E_None; |
891 | if (eScope == page::SCOPE_DIRECT) |
892 | { |
893 | // Truncate all triple indirect pages. |
894 | eErrCode = truncate (page::SCOPE_TRIPLE, 0, rBIOS); |
895 | if (eErrCode != store_E_None) |
896 | return eErrCode; |
897 | |
898 | // Truncate all double indirect pages. |
899 | eErrCode = truncate (page::SCOPE_DOUBLE, 0, rBIOS); |
900 | if (eErrCode != store_E_None) |
901 | return eErrCode; |
902 | |
903 | // Truncate all single indirect pages. |
904 | eErrCode = truncate (page::SCOPE_SINGLE, 0, rBIOS); |
905 | if (eErrCode != store_E_None) |
906 | return eErrCode; |
907 | |
908 | // Truncate direct pages, including 'aLink.m_nIndex0'. |
909 | eErrCode = truncate (eScope, aLink.m_nIndex0, rBIOS); |
910 | } |
911 | else if (eScope == page::SCOPE_SINGLE) |
912 | { |
913 | // Truncate all triple indirect pages. |
914 | eErrCode = truncate (page::SCOPE_TRIPLE, 0, rBIOS); |
915 | if (eErrCode != store_E_None) |
916 | return eErrCode; |
917 | |
918 | // Truncate all double indirect pages. |
919 | eErrCode = truncate (page::SCOPE_DOUBLE, 0, rBIOS); |
920 | if (eErrCode != store_E_None) |
921 | return eErrCode; |
922 | |
923 | // Truncate single indirect pages, downto 'aLink.m_nIndex1'. |
924 | eErrCode = truncate (eScope, aLink.m_nIndex1 + 1, rBIOS); |
925 | if (eErrCode != store_E_None) |
926 | return eErrCode; |
927 | |
928 | // Truncate last single indirect page to ... pages. |
929 | eErrCode = store_truncate_Impl (singleLink (aLink.m_nIndex1), aLink.m_nIndex0, rBIOS); |
930 | if (eErrCode != store_E_None) |
931 | return eErrCode; |
932 | |
933 | // Check for complete truncation. |
934 | if (aLink.m_nIndex0 == 0) |
935 | { |
936 | // Clear pointer to last single indirect page. |
937 | singleLink (aLink.m_nIndex1, STORE_PAGE_NULL((sal_uInt32)(~0))); |
938 | } |
939 | } |
940 | else if (eScope == page::SCOPE_DOUBLE) |
941 | { |
942 | // Truncate all triple indirect pages. |
943 | eErrCode = truncate (page::SCOPE_TRIPLE, 0, rBIOS); |
944 | if (eErrCode != store_E_None) |
945 | return eErrCode; |
946 | |
947 | // Truncate double indirect pages, downto 'aLink.m_nIndex2'. |
948 | eErrCode = truncate (eScope, aLink.m_nIndex2 + 1, rBIOS); |
949 | if (eErrCode != store_E_None) |
950 | return eErrCode; |
951 | |
952 | // Truncate last double indirect page to ... pages. |
953 | eErrCode = store_truncate_Impl ( |
954 | doubleLink (aLink.m_nIndex2), aLink.m_nIndex1, aLink.m_nIndex0, rBIOS); |
955 | if (eErrCode != store_E_None) |
956 | return eErrCode; |
957 | |
958 | // Check for complete truncation. |
959 | if ((aLink.m_nIndex1 + aLink.m_nIndex0) == 0) |
960 | { |
961 | // Clear pointer to last double indirect page. |
962 | doubleLink (aLink.m_nIndex2, STORE_PAGE_NULL((sal_uInt32)(~0))); |
963 | } |
964 | } |
965 | else if (eScope == page::SCOPE_TRIPLE) |
966 | { |
967 | // Truncate triple indirect pages, downto 'aLink.m_nIndex3'. |
968 | eErrCode = truncate (eScope, aLink.m_nIndex3 + 1, rBIOS); |
969 | if (eErrCode != store_E_None) |
970 | return eErrCode; |
971 | |
972 | // Truncate last triple indirect page to ... pages. |
973 | eErrCode = store_truncate_Impl ( |
974 | tripleLink (aLink.m_nIndex3), aLink.m_nIndex2, aLink.m_nIndex1, aLink.m_nIndex0, rBIOS); |
975 | if (eErrCode != store_E_None) |
976 | return eErrCode; |
977 | |
978 | // Check for complete truncation. |
979 | if ((aLink.m_nIndex2 + aLink.m_nIndex1 + aLink.m_nIndex0) == 0) |
980 | { |
981 | // Clear pointer to last triple indirect page. |
982 | tripleLink (aLink.m_nIndex3, STORE_PAGE_NULL((sal_uInt32)(~0))); |
983 | } |
984 | } |
985 | else if (eScope == page::SCOPE_UNREACHABLE) |
986 | { |
987 | // Out of scope. |
988 | eErrCode = store_E_CantSeek; |
989 | } |
990 | else |
991 | { |
992 | // Unknown scope. |
993 | OSL_TRACE("OStoreDirectoryPageObject::put(): scope failed")do { if (true && (1 > 0)) { sal_detail_logFormat(( SAL_DETAIL_LOG_LEVEL_INFO), ("legacy.osl"), ("/usr/local/src/libreoffice/store/source/stordata.cxx" ":" "993" ": "), "OStoreDirectoryPageObject::put(): scope failed" ); } } while (false); |
994 | eErrCode = store_E_Unknown; |
995 | } |
996 | |
997 | // Leave. |
998 | return eErrCode; |
999 | } |
1000 | |
1001 | /* |
1002 | * truncate (external data page scope; private). |
1003 | */ |
1004 | storeError OStoreDirectoryPageObject::truncate ( |
1005 | page::ChunkScope eScope, |
1006 | sal_uInt16 nRemain, |
1007 | OStorePageBIOS &rBIOS) |
1008 | { |
1009 | OStoreDirectoryDataBlock const & rDataBlock = PAGE().m_aDataBlock; |
1010 | |
1011 | // Enter. |
1012 | storeError eErrCode = store_E_None; |
1013 | if (eScope == page::SCOPE_DIRECT) |
1014 | { |
1015 | // Truncate direct data pages. |
1016 | sal_uInt16 i, n = rDataBlock.directCount(); |
1017 | for (i = n; i > nRemain; i--) |
1018 | { |
1019 | // Obtain data page location. |
1020 | sal_uInt32 nAddr = directLink (i - 1); |
1021 | if (nAddr == STORE_PAGE_NULL((sal_uInt32)(~0))) continue; |
1022 | |
1023 | // Free data page. |
1024 | eErrCode = rBIOS.free (nAddr); |
1025 | if (eErrCode != store_E_None) |
1026 | break; |
1027 | |
1028 | // Clear pointer to data page. |
1029 | directLink (i - 1, STORE_PAGE_NULL((sal_uInt32)(~0))); |
1030 | } |
1031 | |
1032 | // Done. |
1033 | return eErrCode; |
1034 | } |
1035 | |
1036 | if (eScope == page::SCOPE_SINGLE) |
1037 | { |
1038 | // Truncate single indirect pages. |
1039 | sal_uInt16 i, n = rDataBlock.singleCount(); |
1040 | for (i = n; i > nRemain; i--) |
1041 | { |
1042 | // Truncate single indirect page to zero data pages. |
1043 | eErrCode = store_truncate_Impl (singleLink (i - 1), 0, rBIOS); |
1044 | if (eErrCode != store_E_None) |
1045 | break; |
1046 | |
1047 | // Clear pointer to single indirect page. |
1048 | singleLink (i - 1, STORE_PAGE_NULL((sal_uInt32)(~0))); |
1049 | } |
1050 | |
1051 | // Done. |
1052 | return eErrCode; |
1053 | } |
1054 | |
1055 | if (eScope == page::SCOPE_DOUBLE) |
1056 | { |
1057 | // Truncate double indirect pages. |
1058 | sal_uInt16 i, n = rDataBlock.doubleCount(); |
1059 | for (i = n; i > nRemain; i--) |
1060 | { |
1061 | // Truncate double indirect page to zero single indirect pages. |
1062 | eErrCode = store_truncate_Impl (doubleLink (i - 1), 0, 0, rBIOS); |
1063 | if (eErrCode != store_E_None) |
1064 | break; |
1065 | |
1066 | // Clear pointer to double indirect page. |
1067 | doubleLink (i - 1, STORE_PAGE_NULL((sal_uInt32)(~0))); |
1068 | } |
1069 | |
1070 | // Done. |
1071 | return eErrCode; |
1072 | } |
1073 | |
1074 | if (eScope == page::SCOPE_TRIPLE) |
1075 | { |
1076 | // Truncate triple indirect pages. |
1077 | sal_uInt16 i, n = rDataBlock.tripleCount(); |
1078 | for (i = n; i > nRemain; i--) |
1079 | { |
1080 | // Truncate to zero double indirect pages. |
1081 | eErrCode = store_truncate_Impl (tripleLink (i - 1), 0, 0, 0, rBIOS); |
1082 | if (eErrCode != store_E_None) |
1083 | break; |
1084 | |
1085 | // Clear pointer to triple indirect page. |
1086 | tripleLink (i - 1, STORE_PAGE_NULL((sal_uInt32)(~0))); |
1087 | } |
1088 | |
1089 | // Done. |
1090 | return eErrCode; |
1091 | } |
1092 | |
1093 | // Invalid scope. |
1094 | return store_E_InvalidAccess; |
1095 | } |
1096 | |
1097 | /* vim:set shiftwidth=4 softtabstop=4 expandtab: */ |