12940b44dSPeter Avalos ///////////////////////////////////////////////////////////////////////////////
22940b44dSPeter Avalos //
32940b44dSPeter Avalos /// \file armthumb.c
42940b44dSPeter Avalos /// \brief Filter for ARM-Thumb binaries
52940b44dSPeter Avalos ///
62940b44dSPeter Avalos // Authors: Igor Pavlov
72940b44dSPeter Avalos // Lasse Collin
82940b44dSPeter Avalos //
92940b44dSPeter Avalos // This file has been put into the public domain.
102940b44dSPeter Avalos // You can do whatever you want with this file.
112940b44dSPeter Avalos //
122940b44dSPeter Avalos ///////////////////////////////////////////////////////////////////////////////
132940b44dSPeter Avalos
142940b44dSPeter Avalos #include "simple_private.h"
152940b44dSPeter Avalos
162940b44dSPeter Avalos
172940b44dSPeter Avalos static size_t
armthumb_code(void * simple lzma_attribute ((__unused__)),uint32_t now_pos,bool is_encoder,uint8_t * buffer,size_t size)1846a2189dSzrj armthumb_code(void *simple lzma_attribute((__unused__)),
192940b44dSPeter Avalos uint32_t now_pos, bool is_encoder,
202940b44dSPeter Avalos uint8_t *buffer, size_t size)
212940b44dSPeter Avalos {
222940b44dSPeter Avalos size_t i;
232940b44dSPeter Avalos for (i = 0; i + 4 <= size; i += 2) {
242940b44dSPeter Avalos if ((buffer[i + 1] & 0xF8) == 0xF0
252940b44dSPeter Avalos && (buffer[i + 3] & 0xF8) == 0xF8) {
26*e151908bSDaniel Fojt uint32_t src = (((uint32_t)(buffer[i + 1]) & 7) << 19)
27*e151908bSDaniel Fojt | ((uint32_t)(buffer[i + 0]) << 11)
28*e151908bSDaniel Fojt | (((uint32_t)(buffer[i + 3]) & 7) << 8)
29*e151908bSDaniel Fojt | (uint32_t)(buffer[i + 2]);
302940b44dSPeter Avalos
312940b44dSPeter Avalos src <<= 1;
322940b44dSPeter Avalos
332940b44dSPeter Avalos uint32_t dest;
342940b44dSPeter Avalos if (is_encoder)
352940b44dSPeter Avalos dest = now_pos + (uint32_t)(i) + 4 + src;
362940b44dSPeter Avalos else
372940b44dSPeter Avalos dest = src - (now_pos + (uint32_t)(i) + 4);
382940b44dSPeter Avalos
392940b44dSPeter Avalos dest >>= 1;
402940b44dSPeter Avalos buffer[i + 1] = 0xF0 | ((dest >> 19) & 0x7);
412940b44dSPeter Avalos buffer[i + 0] = (dest >> 11);
422940b44dSPeter Avalos buffer[i + 3] = 0xF8 | ((dest >> 8) & 0x7);
432940b44dSPeter Avalos buffer[i + 2] = (dest);
442940b44dSPeter Avalos i += 2;
452940b44dSPeter Avalos }
462940b44dSPeter Avalos }
472940b44dSPeter Avalos
482940b44dSPeter Avalos return i;
492940b44dSPeter Avalos }
502940b44dSPeter Avalos
512940b44dSPeter Avalos
522940b44dSPeter Avalos static lzma_ret
armthumb_coder_init(lzma_next_coder * next,const lzma_allocator * allocator,const lzma_filter_info * filters,bool is_encoder)5315ab8c86SJohn Marino armthumb_coder_init(lzma_next_coder *next, const lzma_allocator *allocator,
542940b44dSPeter Avalos const lzma_filter_info *filters, bool is_encoder)
552940b44dSPeter Avalos {
562940b44dSPeter Avalos return lzma_simple_coder_init(next, allocator, filters,
572940b44dSPeter Avalos &armthumb_code, 0, 4, 2, is_encoder);
582940b44dSPeter Avalos }
592940b44dSPeter Avalos
602940b44dSPeter Avalos
612940b44dSPeter Avalos extern lzma_ret
lzma_simple_armthumb_encoder_init(lzma_next_coder * next,const lzma_allocator * allocator,const lzma_filter_info * filters)622940b44dSPeter Avalos lzma_simple_armthumb_encoder_init(lzma_next_coder *next,
6315ab8c86SJohn Marino const lzma_allocator *allocator,
6415ab8c86SJohn Marino const lzma_filter_info *filters)
652940b44dSPeter Avalos {
662940b44dSPeter Avalos return armthumb_coder_init(next, allocator, filters, true);
672940b44dSPeter Avalos }
682940b44dSPeter Avalos
692940b44dSPeter Avalos
702940b44dSPeter Avalos extern lzma_ret
lzma_simple_armthumb_decoder_init(lzma_next_coder * next,const lzma_allocator * allocator,const lzma_filter_info * filters)712940b44dSPeter Avalos lzma_simple_armthumb_decoder_init(lzma_next_coder *next,
7215ab8c86SJohn Marino const lzma_allocator *allocator,
7315ab8c86SJohn Marino const lzma_filter_info *filters)
742940b44dSPeter Avalos {
752940b44dSPeter Avalos return armthumb_coder_init(next, allocator, filters, false);
762940b44dSPeter Avalos }
77