• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2011 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #ifndef ART_LIBARTBASE_BASE_LEB128_H_
18 #define ART_LIBARTBASE_BASE_LEB128_H_
19 
20 #include <vector>
21 
22 #include <android-base/logging.h>
23 
24 #include "bit_utils.h"
25 #include "globals.h"
26 #include "macros.h"
27 
28 namespace art {
29 
30 // Reads an unsigned LEB128 value, updating the given pointer to point
31 // just past the end of the read value. This function tolerates
32 // non-zero high-order bits in the fifth encoded byte.
DecodeUnsignedLeb128(const uint8_t ** data)33 static inline uint32_t DecodeUnsignedLeb128(const uint8_t** data) {
34   const uint8_t* ptr = *data;
35   int result = *(ptr++);
36   if (UNLIKELY(result > 0x7f)) {
37     int cur = *(ptr++);
38     result = (result & 0x7f) | ((cur & 0x7f) << 7);
39     if (cur > 0x7f) {
40       cur = *(ptr++);
41       result |= (cur & 0x7f) << 14;
42       if (cur > 0x7f) {
43         cur = *(ptr++);
44         result |= (cur & 0x7f) << 21;
45         if (cur > 0x7f) {
46           // Note: We don't check to see if cur is out of range here,
47           // meaning we tolerate garbage in the four high-order bits.
48           cur = *(ptr++);
49           result |= cur << 28;
50         }
51       }
52     }
53   }
54   *data = ptr;
55   return static_cast<uint32_t>(result);
56 }
57 
DecodeUnsignedLeb128WithoutMovingCursor(const uint8_t * data)58 static inline uint32_t DecodeUnsignedLeb128WithoutMovingCursor(const uint8_t* data) {
59   return DecodeUnsignedLeb128(&data);
60 }
61 
DecodeUnsignedLeb128Checked(const uint8_t ** data,const void * end,uint32_t * out)62 static inline bool DecodeUnsignedLeb128Checked(const uint8_t** data,
63                                                const void* end,
64                                                uint32_t* out) {
65   const uint8_t* ptr = *data;
66   if (ptr >= end) {
67     return false;
68   }
69   int result = *(ptr++);
70   if (UNLIKELY(result > 0x7f)) {
71     if (ptr >= end) {
72       return false;
73     }
74     int cur = *(ptr++);
75     result = (result & 0x7f) | ((cur & 0x7f) << 7);
76     if (cur > 0x7f) {
77       if (ptr >= end) {
78         return false;
79       }
80       cur = *(ptr++);
81       result |= (cur & 0x7f) << 14;
82       if (cur > 0x7f) {
83         if (ptr >= end) {
84           return false;
85         }
86         cur = *(ptr++);
87         result |= (cur & 0x7f) << 21;
88         if (cur > 0x7f) {
89           if (ptr >= end) {
90             return false;
91           }
92           // Note: We don't check to see if cur is out of range here,
93           // meaning we tolerate garbage in the four high-order bits.
94           cur = *(ptr++);
95           result |= cur << 28;
96         }
97       }
98     }
99   }
100   *data = ptr;
101   *out = static_cast<uint32_t>(result);
102   return true;
103 }
104 
105 // Reads an unsigned LEB128 + 1 value. updating the given pointer to point
106 // just past the end of the read value. This function tolerates
107 // non-zero high-order bits in the fifth encoded byte.
108 // It is possible for this function to return -1.
DecodeUnsignedLeb128P1(const uint8_t ** data)109 static inline int32_t DecodeUnsignedLeb128P1(const uint8_t** data) {
110   return DecodeUnsignedLeb128(data) - 1;
111 }
112 
113 // Reads a signed LEB128 value, updating the given pointer to point
114 // just past the end of the read value. This function tolerates
115 // non-zero high-order bits in the fifth encoded byte.
DecodeSignedLeb128(const uint8_t ** data)116 static inline int32_t DecodeSignedLeb128(const uint8_t** data) {
117   const uint8_t* ptr = *data;
118   int32_t result = *(ptr++);
119   if (result <= 0x7f) {
120     result = (result << 25) >> 25;
121   } else {
122     int cur = *(ptr++);
123     result = (result & 0x7f) | ((cur & 0x7f) << 7);
124     if (cur <= 0x7f) {
125       result = (result << 18) >> 18;
126     } else {
127       cur = *(ptr++);
128       result |= (cur & 0x7f) << 14;
129       if (cur <= 0x7f) {
130         result = (result << 11) >> 11;
131       } else {
132         cur = *(ptr++);
133         result |= (cur & 0x7f) << 21;
134         if (cur <= 0x7f) {
135           result = (result << 4) >> 4;
136         } else {
137           // Note: We don't check to see if cur is out of range here,
138           // meaning we tolerate garbage in the four high-order bits.
139           cur = *(ptr++);
140           result |= cur << 28;
141         }
142       }
143     }
144   }
145   *data = ptr;
146   return result;
147 }
148 
DecodeSignedLeb128Checked(const uint8_t ** data,const void * end,int32_t * out)149 static inline bool DecodeSignedLeb128Checked(const uint8_t** data,
150                                              const void* end,
151                                              int32_t* out) {
152   const uint8_t* ptr = *data;
153   if (ptr >= end) {
154     return false;
155   }
156   int32_t result = *(ptr++);
157   if (result <= 0x7f) {
158     result = (result << 25) >> 25;
159   } else {
160     if (ptr >= end) {
161       return false;
162     }
163     int cur = *(ptr++);
164     result = (result & 0x7f) | ((cur & 0x7f) << 7);
165     if (cur <= 0x7f) {
166       result = (result << 18) >> 18;
167     } else {
168       if (ptr >= end) {
169         return false;
170       }
171       cur = *(ptr++);
172       result |= (cur & 0x7f) << 14;
173       if (cur <= 0x7f) {
174         result = (result << 11) >> 11;
175       } else {
176         if (ptr >= end) {
177           return false;
178         }
179         cur = *(ptr++);
180         result |= (cur & 0x7f) << 21;
181         if (cur <= 0x7f) {
182           result = (result << 4) >> 4;
183         } else {
184           if (ptr >= end) {
185             return false;
186           }
187           // Note: We don't check to see if cur is out of range here,
188           // meaning we tolerate garbage in the four high-order bits.
189           cur = *(ptr++);
190           result |= cur << 28;
191         }
192       }
193     }
194   }
195   *data = ptr;
196   *out = static_cast<uint32_t>(result);
197   return true;
198 }
199 
200 // Returns the number of bytes needed to encode the value in unsigned LEB128.
UnsignedLeb128Size(uint32_t data)201 static inline uint32_t UnsignedLeb128Size(uint32_t data) {
202   // bits_to_encode = (data != 0) ? 32 - CLZ(x) : 1  // 32 - CLZ(data | 1)
203   // bytes = ceil(bits_to_encode / 7.0);             // (6 + bits_to_encode) / 7
204   uint32_t x = 6 + 32 - CLZ(data | 1U);
205   // Division by 7 is done by (x * 37) >> 8 where 37 = ceil(256 / 7).
206   // This works for 0 <= x < 256 / (7 * 37 - 256), i.e. 0 <= x <= 85.
207   return (x * 37) >> 8;
208 }
209 
IsLeb128Terminator(const uint8_t * ptr)210 static inline bool IsLeb128Terminator(const uint8_t* ptr) {
211   return *ptr <= 0x7f;
212 }
213 
214 // Returns the first byte of a Leb128 value assuming that:
215 // (1) `end_ptr` points to the first byte after the Leb128 value, and
216 // (2) there is another Leb128 value before this one.
217 template <typename T>
ReverseSearchUnsignedLeb128(T * end_ptr)218 static inline T* ReverseSearchUnsignedLeb128(T* end_ptr) {
219   static_assert(std::is_same_v<std::remove_const_t<T>, uint8_t>,
220                 "T must be a uint8_t");
221   T* ptr = end_ptr;
222 
223   // Move one byte back, check that this is the terminating byte.
224   ptr--;
225   DCHECK(IsLeb128Terminator(ptr));
226 
227   // Keep moving back while the previous byte is not a terminating byte.
228   // Fail after reading five bytes in case there isn't another Leb128 value
229   // before this one.
230   while (!IsLeb128Terminator(ptr - 1)) {
231     ptr--;
232     DCHECK_LE(static_cast<ptrdiff_t>(end_ptr - ptr), 5);
233   }
234 
235   return ptr;
236 }
237 
238 // Returns the number of bytes needed to encode the value in unsigned LEB128.
SignedLeb128Size(int32_t data)239 static inline uint32_t SignedLeb128Size(int32_t data) {
240   // Like UnsignedLeb128Size(), but we need one bit beyond the highest bit that differs from sign.
241   data = data ^ (data >> 31);
242   uint32_t x = 1 /* we need to encode the sign bit */ + 6 + 32 - CLZ(data | 1U);
243   return (x * 37) >> 8;
244 }
245 
EncodeUnsignedLeb128(uint8_t * dest,uint32_t value)246 static inline uint8_t* EncodeUnsignedLeb128(uint8_t* dest, uint32_t value) {
247   uint8_t out = value & 0x7f;
248   value >>= 7;
249   while (value != 0) {
250     *dest++ = out | 0x80;
251     out = value & 0x7f;
252     value >>= 7;
253   }
254   *dest++ = out;
255   return dest;
256 }
257 
258 template <typename Vector>
EncodeUnsignedLeb128(Vector * dest,uint32_t value)259 static inline void EncodeUnsignedLeb128(Vector* dest, uint32_t value) {
260   static_assert(std::is_same_v<typename Vector::value_type, uint8_t>, "Invalid value type");
261   uint8_t out = value & 0x7f;
262   value >>= 7;
263   while (value != 0) {
264     dest->push_back(out | 0x80);
265     out = value & 0x7f;
266     value >>= 7;
267   }
268   dest->push_back(out);
269 }
270 
271 // Overwrite encoded Leb128 with a new value. The new value must be less than
272 // or equal to the old value to ensure that it fits the allocated space.
UpdateUnsignedLeb128(uint8_t * dest,uint32_t value)273 static inline void UpdateUnsignedLeb128(uint8_t* dest, uint32_t value) {
274   const uint8_t* old_end = dest;
275   uint32_t old_value = DecodeUnsignedLeb128(&old_end);
276   DCHECK_LE(UnsignedLeb128Size(value), UnsignedLeb128Size(old_value));
277   for (uint8_t* end = EncodeUnsignedLeb128(dest, value); end < old_end; end++) {
278     // Use longer encoding than necessary to fill the allocated space.
279     end[-1] |= 0x80;
280     end[0] = 0;
281   }
282 }
283 
EncodeSignedLeb128(uint8_t * dest,int32_t value)284 static inline uint8_t* EncodeSignedLeb128(uint8_t* dest, int32_t value) {
285   uint32_t extra_bits = static_cast<uint32_t>(value ^ (value >> 31)) >> 6;
286   uint8_t out = value & 0x7f;
287   while (extra_bits != 0u) {
288     *dest++ = out | 0x80;
289     value >>= 7;
290     out = value & 0x7f;
291     extra_bits >>= 7;
292   }
293   *dest++ = out;
294   return dest;
295 }
296 
297 template<typename Vector>
EncodeSignedLeb128(Vector * dest,int32_t value)298 static inline void EncodeSignedLeb128(Vector* dest, int32_t value) {
299   static_assert(std::is_same_v<typename Vector::value_type, uint8_t>, "Invalid value type");
300   uint32_t extra_bits = static_cast<uint32_t>(value ^ (value >> 31)) >> 6;
301   uint8_t out = value & 0x7f;
302   while (extra_bits != 0u) {
303     dest->push_back(out | 0x80);
304     value >>= 7;
305     out = value & 0x7f;
306     extra_bits >>= 7;
307   }
308   dest->push_back(out);
309 }
310 
311 // An encoder that pushes int32_t/uint32_t data onto the given std::vector.
312 template <typename Vector = std::vector<uint8_t>>
313 class Leb128Encoder {
314   static_assert(std::is_same_v<typename Vector::value_type, uint8_t>, "Invalid value type");
315 
316  public:
Leb128Encoder(Vector * data)317   explicit Leb128Encoder(Vector* data) : data_(data) {
318     DCHECK(data != nullptr);
319   }
320 
Reserve(uint32_t size)321   void Reserve(uint32_t size) {
322     data_->reserve(size);
323   }
324 
PushBackUnsigned(uint32_t value)325   void PushBackUnsigned(uint32_t value) {
326     EncodeUnsignedLeb128(data_, value);
327   }
328 
329   template<typename It>
InsertBackUnsigned(It cur,It end)330   void InsertBackUnsigned(It cur, It end) {
331     for (; cur != end; ++cur) {
332       PushBackUnsigned(*cur);
333     }
334   }
335 
PushBackSigned(int32_t value)336   void PushBackSigned(int32_t value) {
337     EncodeSignedLeb128(data_, value);
338   }
339 
340   template<typename It>
InsertBackSigned(It cur,It end)341   void InsertBackSigned(It cur, It end) {
342     for (; cur != end; ++cur) {
343       PushBackSigned(*cur);
344     }
345   }
346 
GetData()347   const Vector& GetData() const {
348     return *data_;
349   }
350 
351  protected:
352   Vector* const data_;
353 
354  private:
355   DISALLOW_COPY_AND_ASSIGN(Leb128Encoder);
356 };
357 
358 // An encoder with an API similar to vector<uint32_t> where the data is captured in ULEB128 format.
359 template <typename Vector = std::vector<uint8_t>>
360 class Leb128EncodingVector final : private Vector,
361                                    public Leb128Encoder<Vector> {
362   static_assert(std::is_same_v<typename Vector::value_type, uint8_t>, "Invalid value type");
363 
364  public:
Leb128EncodingVector()365   Leb128EncodingVector() : Leb128Encoder<Vector>(this) { }
366 
Leb128EncodingVector(const typename Vector::allocator_type & alloc)367   explicit Leb128EncodingVector(const typename Vector::allocator_type& alloc)
368     : Vector(alloc),
369       Leb128Encoder<Vector>(this) { }
370 
371  private:
372   DISALLOW_COPY_AND_ASSIGN(Leb128EncodingVector);
373 };
374 
375 }  // namespace art
376 
377 #endif  // ART_LIBARTBASE_BASE_LEB128_H_
378