1*fb1b10abSAndroid Build Coastguard Worker#!/usr/bin/env perl 2*fb1b10abSAndroid Build Coastguard Worker## 3*fb1b10abSAndroid Build Coastguard Worker## Copyright (c) 2013 The WebM project authors. All Rights Reserved. 4*fb1b10abSAndroid Build Coastguard Worker## 5*fb1b10abSAndroid Build Coastguard Worker## Use of this source code is governed by a BSD-style license 6*fb1b10abSAndroid Build Coastguard Worker## that can be found in the LICENSE file in the root of the source 7*fb1b10abSAndroid Build Coastguard Worker## tree. An additional intellectual property rights grant can be found 8*fb1b10abSAndroid Build Coastguard Worker## in the file PATENTS. All contributing project authors may 9*fb1b10abSAndroid Build Coastguard Worker## be found in the AUTHORS file in the root of the source tree. 10*fb1b10abSAndroid Build Coastguard Worker## 11*fb1b10abSAndroid Build Coastguard Worker 12*fb1b10abSAndroid Build Coastguard Workerpackage thumb; 13*fb1b10abSAndroid Build Coastguard Worker 14*fb1b10abSAndroid Build Coastguard Workersub FixThumbInstructions($) 15*fb1b10abSAndroid Build Coastguard Worker{ 16*fb1b10abSAndroid Build Coastguard Worker # Write additions with shifts, such as "add r10, r11, lsl #8", 17*fb1b10abSAndroid Build Coastguard Worker # in three operand form, "add r10, r10, r11, lsl #8". 18*fb1b10abSAndroid Build Coastguard Worker s/(add\s+)(r\d+),\s*(r\d+),\s*(lsl #\d+)/$1$2, $2, $3, $4/g; 19*fb1b10abSAndroid Build Coastguard Worker 20*fb1b10abSAndroid Build Coastguard Worker # Convert additions with a non-constant shift into a sequence 21*fb1b10abSAndroid Build Coastguard Worker # with left shift, addition and a right shift (to restore the 22*fb1b10abSAndroid Build Coastguard Worker # register to the original value). Currently the right shift 23*fb1b10abSAndroid Build Coastguard Worker # isn't necessary in the code base since the values in these 24*fb1b10abSAndroid Build Coastguard Worker # registers aren't used, but doing the shift for consistency. 25*fb1b10abSAndroid Build Coastguard Worker # This converts instructions such as "add r12, r12, r5, lsl r4" 26*fb1b10abSAndroid Build Coastguard Worker # into the sequence "lsl r5, r4", "add r12, r12, r5", "lsr r5, r4". 27*fb1b10abSAndroid Build Coastguard Worker s/^(\s*)(add)(\s+)(r\d+),\s*(r\d+),\s*(r\d+),\s*lsl (r\d+)/$1lsl$3$6, $7\n$1$2$3$4, $5, $6\n$1lsr$3$6, $7/g; 28*fb1b10abSAndroid Build Coastguard Worker 29*fb1b10abSAndroid Build Coastguard Worker # Convert loads with right shifts in the indexing into a 30*fb1b10abSAndroid Build Coastguard Worker # sequence of an add, load and sub. This converts 31*fb1b10abSAndroid Build Coastguard Worker # "ldrb r4, [r9, lr, asr #1]" into "add r9, r9, lr, asr #1", 32*fb1b10abSAndroid Build Coastguard Worker # "ldrb r9, [r9]", "sub r9, r9, lr, asr #1". 33*fb1b10abSAndroid Build Coastguard Worker s/^(\s*)(ldrb)(\s+)(r\d+),\s*\[(\w+),\s*(\w+),\s*(asr #\d+)\]/$1add $3$5, $5, $6, $7\n$1$2$3$4, [$5]\n$1sub $3$5, $5, $6, $7/g; 34*fb1b10abSAndroid Build Coastguard Worker 35*fb1b10abSAndroid Build Coastguard Worker # Convert register indexing with writeback into a separate add 36*fb1b10abSAndroid Build Coastguard Worker # instruction. This converts "ldrb r12, [r1, r2]!" into 37*fb1b10abSAndroid Build Coastguard Worker # "ldrb r12, [r1, r2]", "add r1, r1, r2". 38*fb1b10abSAndroid Build Coastguard Worker s/^(\s*)(ldrb)(\s+)(r\d+),\s*\[(\w+),\s*(\w+)\]!/$1$2$3$4, [$5, $6]\n$1add $3$5, $6/g; 39*fb1b10abSAndroid Build Coastguard Worker 40*fb1b10abSAndroid Build Coastguard Worker # Convert negative register indexing into separate sub/add instructions. 41*fb1b10abSAndroid Build Coastguard Worker # This converts "ldrne r4, [src, -pstep, lsl #1]" into 42*fb1b10abSAndroid Build Coastguard Worker # "subne src, src, pstep, lsl #1", "ldrne r4, [src]", 43*fb1b10abSAndroid Build Coastguard Worker # "addne src, src, pstep, lsl #1". In a couple of cases where 44*fb1b10abSAndroid Build Coastguard Worker # this is used, it's used for two subsequent load instructions, 45*fb1b10abSAndroid Build Coastguard Worker # where a hand-written version of it could merge two subsequent 46*fb1b10abSAndroid Build Coastguard Worker # add and sub instructions. 47*fb1b10abSAndroid Build Coastguard Worker s/^(\s*)((ldr|str|pld)(ne)?)(\s+)(r\d+,\s*)?\[(\w+), -([^\]]+)\]/$1sub$4$5$7, $7, $8\n$1$2$5$6\[$7\]\n$1add$4$5$7, $7, $8/g; 48*fb1b10abSAndroid Build Coastguard Worker 49*fb1b10abSAndroid Build Coastguard Worker # Convert register post indexing to a separate add instruction. 50*fb1b10abSAndroid Build Coastguard Worker # This converts "ldrneb r9, [r0], r2" into "ldrneb r9, [r0]", 51*fb1b10abSAndroid Build Coastguard Worker # "addne r0, r0, r2". 52*fb1b10abSAndroid Build Coastguard Worker s/^(\s*)((ldr|str)(ne)?[bhd]?)(\s+)(\w+),(\s*\w+,)?\s*\[(\w+)\],\s*(\w+)/$1$2$5$6,$7 [$8]\n$1add$4$5$8, $8, $9/g; 53*fb1b10abSAndroid Build Coastguard Worker 54*fb1b10abSAndroid Build Coastguard Worker # Convert "mov pc, lr" into "bx lr", since the former only works 55*fb1b10abSAndroid Build Coastguard Worker # for switching from arm to thumb (and only in armv7), but not 56*fb1b10abSAndroid Build Coastguard Worker # from thumb to arm. 57*fb1b10abSAndroid Build Coastguard Worker s/mov(\s*)pc\s*,\s*lr/bx$1lr/g; 58*fb1b10abSAndroid Build Coastguard Worker} 59*fb1b10abSAndroid Build Coastguard Worker 60*fb1b10abSAndroid Build Coastguard Worker1; 61