Encoder.cpp 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206
  1. /*
  2. * Copyright (c) 2020, Andreas Kling <kling@serenityos.org>
  3. * Copyright (c) 2021, kleines Filmröllchen <malu.bertsch@gmail.com>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include <AK/BitCast.h>
  8. #include <AK/ByteBuffer.h>
  9. #include <AK/String.h>
  10. #include <AK/URL.h>
  11. #include <LibCore/AnonymousBuffer.h>
  12. #include <LibCore/DateTime.h>
  13. #include <LibIPC/Dictionary.h>
  14. #include <LibIPC/Encoder.h>
  15. #include <LibIPC/File.h>
  16. namespace IPC {
  17. Encoder& Encoder::operator<<(bool value)
  18. {
  19. return *this << (u8)value;
  20. }
  21. Encoder& Encoder::operator<<(u8 value)
  22. {
  23. m_buffer.data.append(value);
  24. return *this;
  25. }
  26. Encoder& Encoder::operator<<(u16 value)
  27. {
  28. m_buffer.data.ensure_capacity(m_buffer.data.size() + 2);
  29. m_buffer.data.unchecked_append((u8)value);
  30. m_buffer.data.unchecked_append((u8)(value >> 8));
  31. return *this;
  32. }
  33. void Encoder::encode_u32(u32 value)
  34. {
  35. m_buffer.data.ensure_capacity(m_buffer.data.size() + 4);
  36. m_buffer.data.unchecked_append((u8)value);
  37. m_buffer.data.unchecked_append((u8)(value >> 8));
  38. m_buffer.data.unchecked_append((u8)(value >> 16));
  39. m_buffer.data.unchecked_append((u8)(value >> 24));
  40. }
  41. void Encoder::encode_u64(u64 value)
  42. {
  43. m_buffer.data.ensure_capacity(m_buffer.data.size() + 8);
  44. m_buffer.data.unchecked_append((u8)value);
  45. m_buffer.data.unchecked_append((u8)(value >> 8));
  46. m_buffer.data.unchecked_append((u8)(value >> 16));
  47. m_buffer.data.unchecked_append((u8)(value >> 24));
  48. m_buffer.data.unchecked_append((u8)(value >> 32));
  49. m_buffer.data.unchecked_append((u8)(value >> 40));
  50. m_buffer.data.unchecked_append((u8)(value >> 48));
  51. m_buffer.data.unchecked_append((u8)(value >> 56));
  52. }
  53. Encoder& Encoder::operator<<(unsigned value)
  54. {
  55. encode_u32(value);
  56. return *this;
  57. }
  58. Encoder& Encoder::operator<<(unsigned long value)
  59. {
  60. if constexpr (sizeof(value) == 4)
  61. encode_u32(value);
  62. else
  63. encode_u64(value);
  64. return *this;
  65. }
  66. Encoder& Encoder::operator<<(unsigned long long value)
  67. {
  68. if constexpr (sizeof(value) == 4)
  69. encode_u32(value);
  70. else
  71. encode_u64(value);
  72. return *this;
  73. }
  74. Encoder& Encoder::operator<<(i8 value)
  75. {
  76. m_buffer.data.append((u8)value);
  77. return *this;
  78. }
  79. Encoder& Encoder::operator<<(i16 value)
  80. {
  81. m_buffer.data.ensure_capacity(m_buffer.data.size() + 2);
  82. m_buffer.data.unchecked_append((u8)value);
  83. m_buffer.data.unchecked_append((u8)(value >> 8));
  84. return *this;
  85. }
  86. Encoder& Encoder::operator<<(i32 value)
  87. {
  88. m_buffer.data.ensure_capacity(m_buffer.data.size() + 4);
  89. m_buffer.data.unchecked_append((u8)value);
  90. m_buffer.data.unchecked_append((u8)(value >> 8));
  91. m_buffer.data.unchecked_append((u8)(value >> 16));
  92. m_buffer.data.unchecked_append((u8)(value >> 24));
  93. return *this;
  94. }
  95. Encoder& Encoder::operator<<(i64 value)
  96. {
  97. m_buffer.data.ensure_capacity(m_buffer.data.size() + 8);
  98. m_buffer.data.unchecked_append((u8)value);
  99. m_buffer.data.unchecked_append((u8)(value >> 8));
  100. m_buffer.data.unchecked_append((u8)(value >> 16));
  101. m_buffer.data.unchecked_append((u8)(value >> 24));
  102. m_buffer.data.unchecked_append((u8)(value >> 32));
  103. m_buffer.data.unchecked_append((u8)(value >> 40));
  104. m_buffer.data.unchecked_append((u8)(value >> 48));
  105. m_buffer.data.unchecked_append((u8)(value >> 56));
  106. return *this;
  107. }
  108. Encoder& Encoder::operator<<(float value)
  109. {
  110. u32 as_u32 = bit_cast<u32>(value);
  111. return *this << as_u32;
  112. }
  113. Encoder& Encoder::operator<<(double value)
  114. {
  115. u64 as_u64 = bit_cast<u64>(value);
  116. return *this << as_u64;
  117. }
  118. Encoder& Encoder::operator<<(char const* value)
  119. {
  120. return *this << StringView(value);
  121. }
  122. Encoder& Encoder::operator<<(StringView value)
  123. {
  124. m_buffer.data.append((u8 const*)value.characters_without_null_termination(), value.length());
  125. return *this;
  126. }
  127. Encoder& Encoder::operator<<(String const& value)
  128. {
  129. if (value.is_null())
  130. return *this << (i32)-1;
  131. *this << static_cast<i32>(value.length());
  132. return *this << value.view();
  133. }
  134. Encoder& Encoder::operator<<(ByteBuffer const& value)
  135. {
  136. *this << static_cast<i32>(value.size());
  137. m_buffer.data.append(value.data(), value.size());
  138. return *this;
  139. }
  140. Encoder& Encoder::operator<<(URL const& value)
  141. {
  142. return *this << value.to_string();
  143. }
  144. Encoder& Encoder::operator<<(Dictionary const& dictionary)
  145. {
  146. *this << (u64)dictionary.size();
  147. dictionary.for_each_entry([this](auto& key, auto& value) {
  148. *this << key << value;
  149. });
  150. return *this;
  151. }
  152. Encoder& Encoder::operator<<(File const& file)
  153. {
  154. int fd = file.fd();
  155. if (fd != -1) {
  156. auto result = dup(fd);
  157. if (result < 0) {
  158. perror("dup");
  159. VERIFY_NOT_REACHED();
  160. }
  161. fd = result;
  162. }
  163. m_buffer.fds.append(adopt_ref(*new AutoCloseFileDescriptor(fd)));
  164. return *this;
  165. }
  166. bool encode(Encoder& encoder, Core::AnonymousBuffer const& buffer)
  167. {
  168. encoder << buffer.is_valid();
  169. if (buffer.is_valid()) {
  170. encoder << (u32)buffer.size();
  171. encoder << IPC::File(buffer.fd());
  172. }
  173. return true;
  174. }
  175. bool encode(Encoder& encoder, Core::DateTime const& datetime)
  176. {
  177. encoder << static_cast<i64>(datetime.timestamp());
  178. return true;
  179. }
  180. }