sql-make-virtualcursor-standard-layout-type.patch 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. From 80368f8ba7a8bab13440463a254888311efe3986 Mon Sep 17 00:00:00 2001
  2. From: Stephan Hartmann <stha09@googlemail.com>
  3. Date: Tue, 04 May 2021 15:00:19 +0000
  4. Subject: [PATCH] sql: make VirtualCursor standard layout type
  5. sql::recover::VirtualCursor needs to be a standard layout type, but
  6. has members of type std::unique_ptr. However, std::unique_ptr is not
  7. guaranteed to be standard layout. Compiling with clang combined with
  8. gcc-11 libstdc++ fails because of this. Replace std::unique_ptr with
  9. raw pointers.
  10. Bug: 1189788
  11. Change-Id: Ia6dc388cc5ef1c0f2afc75f8ca45b9f12687ca9c
  12. ---
  13. diff --git a/sql/recover_module/btree.cc b/sql/recover_module/btree.cc
  14. index 9ecaafe..839318a 100644
  15. --- a/sql/recover_module/btree.cc
  16. +++ b/sql/recover_module/btree.cc
  17. @@ -135,16 +135,25 @@
  18. "Move the destructor to the .cc file if it's non-trival");
  19. #endif // !DCHECK_IS_ON()
  20. -LeafPageDecoder::LeafPageDecoder(DatabasePageReader* db_reader) noexcept
  21. - : page_id_(db_reader->page_id()),
  22. - db_reader_(db_reader),
  23. - cell_count_(ComputeCellCount(db_reader)),
  24. - next_read_index_(0),
  25. - last_record_size_(0) {
  26. +void LeafPageDecoder::Initialize(DatabasePageReader* db_reader) {
  27. + DCHECK(db_reader);
  28. DCHECK(IsOnValidPage(db_reader));
  29. + page_id_ = db_reader->page_id();
  30. + db_reader_ = db_reader;
  31. + cell_count_ = ComputeCellCount(db_reader);
  32. + next_read_index_ = 0;
  33. + last_record_size_ = 0;
  34. DCHECK(DatabasePageReader::IsValidPageId(page_id_));
  35. }
  36. +void LeafPageDecoder::Reset() {
  37. + db_reader_ = nullptr;
  38. + page_id_ = 0;
  39. + cell_count_ = 0;
  40. + next_read_index_ = 0;
  41. + last_record_size_ = 0;
  42. +}
  43. +
  44. bool LeafPageDecoder::TryAdvance() {
  45. DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
  46. DCHECK(CanAdvance());
  47. diff --git a/sql/recover_module/btree.h b/sql/recover_module/btree.h
  48. index d76d076..33114b0 100644
  49. --- a/sql/recover_module/btree.h
  50. +++ b/sql/recover_module/btree.h
  51. @@ -102,7 +102,7 @@
  52. //
  53. // |db_reader| must have been used to read an inner page of a table B-tree.
  54. // |db_reader| must outlive this instance.
  55. - explicit LeafPageDecoder(DatabasePageReader* db_reader) noexcept;
  56. + explicit LeafPageDecoder() noexcept = default;
  57. ~LeafPageDecoder() noexcept = default;
  58. LeafPageDecoder(const LeafPageDecoder&) = delete;
  59. @@ -150,6 +150,15 @@
  60. // read as long as CanAdvance() returns true.
  61. bool TryAdvance();
  62. + // Initialize with DatabasePageReader
  63. + void Initialize(DatabasePageReader* db_reader);
  64. +
  65. + // Reset internal DatabasePageReader
  66. + void Reset();
  67. +
  68. + // True if DatabasePageReader is valid
  69. + bool IsValid() { return (db_reader_ != nullptr); }
  70. +
  71. // True if the given reader may point to an inner page in a table B-tree.
  72. //
  73. // The last ReadPage() call on |db_reader| must have succeeded.
  74. @@ -163,14 +172,14 @@
  75. static int ComputeCellCount(DatabasePageReader* db_reader);
  76. // The number of the B-tree page this reader is reading.
  77. - const int64_t page_id_;
  78. + int64_t page_id_;
  79. // Used to read the tree page.
  80. //
  81. // Raw pointer usage is acceptable because this instance's owner is expected
  82. // to ensure that the DatabasePageReader outlives this.
  83. - DatabasePageReader* const db_reader_;
  84. + DatabasePageReader* db_reader_;
  85. // Caches the ComputeCellCount() value for this reader's page.
  86. - const int cell_count_ = ComputeCellCount(db_reader_);
  87. + int cell_count_;
  88. // The reader's cursor state.
  89. //
  90. diff --git a/sql/recover_module/cursor.cc b/sql/recover_module/cursor.cc
  91. index 0029ff9..42548bc 100644
  92. --- a/sql/recover_module/cursor.cc
  93. +++ b/sql/recover_module/cursor.cc
  94. @@ -26,7 +26,7 @@
  95. int VirtualCursor::First() {
  96. DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
  97. inner_decoders_.clear();
  98. - leaf_decoder_ = nullptr;
  99. + leaf_decoder_.Reset();
  100. AppendPageDecoder(table_->root_page_id());
  101. return Next();
  102. @@ -36,18 +36,18 @@
  103. DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
  104. record_reader_.Reset();
  105. - while (!inner_decoders_.empty() || leaf_decoder_.get()) {
  106. - if (leaf_decoder_.get()) {
  107. - if (!leaf_decoder_->CanAdvance()) {
  108. + while (!inner_decoders_.empty() || leaf_decoder_.IsValid()) {
  109. + if (leaf_decoder_.IsValid()) {
  110. + if (!leaf_decoder_.CanAdvance()) {
  111. // The leaf has been exhausted. Remove it from the DFS stack.
  112. - leaf_decoder_ = nullptr;
  113. + leaf_decoder_.Reset();
  114. continue;
  115. }
  116. - if (!leaf_decoder_->TryAdvance())
  117. + if (!leaf_decoder_.TryAdvance())
  118. continue;
  119. - if (!payload_reader_.Initialize(leaf_decoder_->last_record_size(),
  120. - leaf_decoder_->last_record_offset())) {
  121. + if (!payload_reader_.Initialize(leaf_decoder_.last_record_size(),
  122. + leaf_decoder_.last_record_offset())) {
  123. continue;
  124. }
  125. if (!record_reader_.Initialize())
  126. @@ -99,13 +99,13 @@
  127. int64_t VirtualCursor::RowId() {
  128. DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
  129. DCHECK(record_reader_.IsInitialized());
  130. - DCHECK(leaf_decoder_.get());
  131. - return leaf_decoder_->last_record_rowid();
  132. + DCHECK(leaf_decoder_.IsValid());
  133. + return leaf_decoder_.last_record_rowid();
  134. }
  135. void VirtualCursor::AppendPageDecoder(int page_id) {
  136. DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
  137. - DCHECK(leaf_decoder_.get() == nullptr)
  138. + DCHECK(!leaf_decoder_.IsValid())
  139. << __func__
  140. << " must only be called when the current path has no leaf decoder";
  141. @@ -113,7 +113,7 @@
  142. return;
  143. if (LeafPageDecoder::IsOnValidPage(&db_reader_)) {
  144. - leaf_decoder_ = std::make_unique<LeafPageDecoder>(&db_reader_);
  145. + leaf_decoder_.Initialize(&db_reader_);
  146. return;
  147. }
  148. diff --git a/sql/recover_module/cursor.h b/sql/recover_module/cursor.h
  149. index afcd690..b15c31d 100644
  150. --- a/sql/recover_module/cursor.h
  151. +++ b/sql/recover_module/cursor.h
  152. @@ -129,7 +129,7 @@
  153. std::vector<std::unique_ptr<InnerPageDecoder>> inner_decoders_;
  154. // Decodes the leaf page containing records.
  155. - std::unique_ptr<LeafPageDecoder> leaf_decoder_;
  156. + LeafPageDecoder leaf_decoder_;
  157. SEQUENCE_CHECKER(sequence_checker_);
  158. };
  159. diff --git a/sql/recover_module/pager.cc b/sql/recover_module/pager.cc
  160. index 58e75de..5fe9620 100644
  161. --- a/sql/recover_module/pager.cc
  162. +++ b/sql/recover_module/pager.cc
  163. @@ -23,8 +23,7 @@
  164. "ints are not appropriate for representing page IDs");
  165. DatabasePageReader::DatabasePageReader(VirtualTable* table)
  166. - : page_data_(std::make_unique<uint8_t[]>(table->page_size())),
  167. - table_(table) {
  168. + : page_data_(), table_(table) {
  169. DCHECK(table != nullptr);
  170. DCHECK(IsValidPageSize(table->page_size()));
  171. }
  172. @@ -57,8 +56,8 @@
  173. std::numeric_limits<int64_t>::max(),
  174. "The |read_offset| computation above may overflow");
  175. - int sqlite_status =
  176. - RawRead(sqlite_file, read_size, read_offset, page_data_.get());
  177. + int sqlite_status = RawRead(sqlite_file, read_size, read_offset,
  178. + const_cast<uint8_t*>(page_data_.data()));
  179. // |page_id_| needs to be set to kInvalidPageId if the read failed.
  180. // Otherwise, future ReadPage() calls with the previous |page_id_| value
  181. diff --git a/sql/recover_module/pager.h b/sql/recover_module/pager.h
  182. index 0e388ddc..99314e3 100644
  183. --- a/sql/recover_module/pager.h
  184. +++ b/sql/recover_module/pager.h
  185. @@ -5,6 +5,7 @@
  186. #ifndef SQL_RECOVER_MODULE_PAGER_H_
  187. #define SQL_RECOVER_MODULE_PAGER_H_
  188. +#include <array>
  189. #include <cstdint>
  190. #include <memory>
  191. @@ -70,7 +71,7 @@
  192. DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
  193. DCHECK_NE(page_id_, kInvalidPageId)
  194. << "Successful ReadPage() required before accessing pager state";
  195. - return page_data_.get();
  196. + return page_data_.data();
  197. }
  198. // The number of bytes in the page read by the last ReadPage() call.
  199. @@ -137,7 +138,7 @@
  200. int page_id_ = kInvalidPageId;
  201. // Stores the bytes of the last page successfully read by ReadPage().
  202. // The content is undefined if the last call to ReadPage() did not succeed.
  203. - const std::unique_ptr<uint8_t[]> page_data_;
  204. + const std::array<uint8_t, kMaxPageSize> page_data_;
  205. // Raw pointer usage is acceptable because this instance's owner is expected
  206. // to ensure that the VirtualTable outlives this.
  207. VirtualTable* const table_;