virtual_binary.cpp 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. /*
  2. XCC Utilities and Library
  3. Copyright (C) 2001 Olaf van der Spek <[email protected]>
  4. This program is free software: you can redistribute it and/or modify
  5. it under the terms of the GNU General Public License as published by
  6. the Free Software Foundation, either version 3 of the License, or
  7. (at your option) any later version.
  8. This program is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. GNU General Public License for more details.
  12. You should have received a copy of the GNU General Public License
  13. along with this program. If not, see <https://www.gnu.org/licenses/>.
  14. */
  15. #include "stdafx.h"
  16. #include <virtual_binary.h>
  17. #include "file32.h"
  18. Cvirtual_binary_source::Cvirtual_binary_source(const void* d, size_t cb_d, const std::shared_ptr<const void>& source)
  19. {
  20. if (source)
  21. {
  22. m_data = const_cast<byte*>(reinterpret_cast<const byte*>(d));
  23. m_size = cb_d;
  24. m_source = source;
  25. }
  26. else
  27. {
  28. m_data = new byte[cb_d];
  29. m_size = cb_d;
  30. if (d)
  31. memcpy(m_data, d, cb_d);
  32. }
  33. mc_references = 1;
  34. }
  35. Cvirtual_binary_source* Cvirtual_binary_source::attach()
  36. {
  37. if (this)
  38. mc_references++;
  39. return this;
  40. }
  41. void Cvirtual_binary_source::detach()
  42. {
  43. if (!this || --mc_references)
  44. return;
  45. if (!m_source)
  46. delete[] m_data;
  47. delete this;
  48. }
  49. Cvirtual_binary_source* Cvirtual_binary_source::pre_edit()
  50. {
  51. if (mc_references == 1 && !m_source)
  52. return this;
  53. Cvirtual_binary_source* t = new Cvirtual_binary_source(m_data, m_size, NULL);
  54. detach();
  55. return t;
  56. }
  57. Cvirtual_binary::Cvirtual_binary()
  58. {
  59. m_source = NULL;
  60. }
  61. Cvirtual_binary::Cvirtual_binary(const Cvirtual_binary& v)
  62. {
  63. m_source = v.m_source->attach();
  64. }
  65. Cvirtual_binary::Cvirtual_binary(const void* d, size_t cb_d)
  66. {
  67. m_source = new Cvirtual_binary_source(d, cb_d);
  68. }
  69. Cvirtual_binary::Cvirtual_binary(const void* d, size_t cb_d, const std::shared_ptr<void>& source)
  70. {
  71. m_source = new Cvirtual_binary_source(d, cb_d, source);
  72. }
  73. Cvirtual_binary::Cvirtual_binary(data_ref d)
  74. {
  75. m_source = new Cvirtual_binary_source(d.data(), d.size());
  76. }
  77. Cvirtual_binary::Cvirtual_binary(const string& fname, bool use_mm)
  78. {
  79. m_source = NULL;
  80. load(fname, use_mm);
  81. }
  82. Cvirtual_binary::~Cvirtual_binary()
  83. {
  84. m_source->detach();
  85. }
  86. const Cvirtual_binary& Cvirtual_binary::operator=(const Cvirtual_binary& v)
  87. {
  88. if (this != &v)
  89. {
  90. m_source->detach();
  91. m_source = v.m_source->attach();
  92. }
  93. return *this;
  94. }
  95. int Cvirtual_binary::save(const string& fname) const
  96. {
  97. return data() ? file32_write(fname, data(), size()) : 1;
  98. }
  99. int Cvirtual_binary::load(const string& fname, bool use_mm)
  100. {
  101. if (use_mm)
  102. *this = file32_read(fname);
  103. else
  104. {
  105. Cvirtual_binary d = file32_read(fname);
  106. *this = Cvirtual_binary(d.data(), d.size());
  107. }
  108. return !data();
  109. }
  110. void Cvirtual_binary::clear()
  111. {
  112. m_source->detach();
  113. m_source = NULL;
  114. }
  115. void Cvirtual_binary::memset(int v)
  116. {
  117. ::memset(data_edit(), v, size());
  118. }
  119. size_t Cvirtual_binary::read(void* d) const
  120. {
  121. memcpy(d, data(), size());
  122. return size();
  123. }
  124. byte* Cvirtual_binary::write_start(size_t cb_d)
  125. {
  126. if (data() && size() == cb_d)
  127. return data_edit();
  128. m_source->detach();
  129. m_source = new Cvirtual_binary_source(NULL, cb_d);
  130. return data_edit();
  131. }
  132. void Cvirtual_binary::write(const void* d, size_t cb_d)
  133. {
  134. memcpy(write_start(cb_d), d, cb_d);
  135. }
  136. Cvirtual_binary Cvirtual_binary::sub_bin(size_t offset, size_t size) const
  137. {
  138. assert(offset >= 0 && offset + size <= Cvirtual_binary::size());
  139. return data() ? Cvirtual_binary(data() + offset, size, std::make_shared<Cvirtual_binary>(*this)) : *this;
  140. }