0x00 Introduction

In this paper, a formal algorithm for taint analysis is defined by Operational Semantics of SIMPIL language, which can be transformed to from languages like C. However, it is possible that some of the rules may create incorrect result. Note: this ariticle does not garantee to be academically professional and is only for my own notes and study, so you may not like it very much.

0x01 T-LOAD

1548465137086

As shown, whether the evaluation result is tainted depends on the return value of Pmem, which is typically defined as Pmem(ta, tv) = tv according to the Table 3 in the paper. That is saying we only consider whether the source memory at the given address is tainted, ignoring the taint state of address value. However, this cause problem when we use a “map table” to convert the form of data. For example, in base64 encode, we may need a utility like this

char map_to_base64_char(uint8_t val)
{//pre:0<=val<64
    char tab[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
    return tab[val];
}

Since the a constant string is assigned to tab, the contents in tab must be untainted. According to Pmem(ta, tv) = tv, the tab[val] is also untainted, which is clearly not the case: tab[val] is base64 form for val, so if val is tainted, tab[val] should also be tainted!

Then, the paper also suggested to change the definition to Pmem(ta, tv) = ta ∨ tv? However, we can still let it produce wrong result. Consider this code

char return_A(size_t some_tainted_data)
{
    char a[0x100];
    size_t still_tainted = some_tainted_data % 0x100;
    memset(a, 'A', 0x100);
    return a[still_tainted];
}

For this function, whatever the some_tainted_data is, the function always returns a constant 'A'. However, the rule Pmem(ta, tv) = ta ∨ tv will say that a[still_tainted] is tainted, which is not what we want. Fortunately this kind of codes rarely occurs unless the developer has intentionally added some anti reverse engineering obfuscation codes.

Another way to fix this issue is to record the address or contents of a specific mapping table, and denote it as a function. Only if the address or content matches the table we have recorded do we consider ta as a taint contributor. However, this requires information about both table start and index, which cannot be represented by semantics of SIMPIL. But in other kinds of instruction set such as x86, the compiler will generate the code like this when we use mapping table: mov dst_reg, [mapping_tab + index * size_of_element], where mapping_tab can be both immediate number of register, index is usually a register, and size_of_element is a constant that can be power of 2. Using this feature, we may redefine it as Pmem(tat, tai, tv) = if_mapping_table(tat) ? tai : tv, where tat stands for address of mapping table and tai stands for the index.

And how do we define if_mapping_table? As I said, we can define it using address itself or content within address, or both. We may just use the value of address, since usually the mapping table is in global variable. But what if not? Like the case of map_to_base64_char above. From my perspective, using content must be an effective way but it is slow, so we can check the address first, then secondly check content if address does not match to know addresses.

0x02 T-BINOP

1548620761843

Pbinop(t1, t2) = t1 ∨ t2, and Pbincheck always returns true according to his definition.

This is saying, the taint state of the return value of binary operator is an or operation of taint states of 2 input values. This will be problematic in some cases.

int tainted1 tainted2;
//...
int tainted_wrong1 = tainted1 - tainted1;
int tainted_wrong2 = tainted1 ^ tainted1;

As shown above, it is obvious that 2 tainted_wrong variables are 0 always, but the semantics above will identify them as tainted. The paper indeed pointed this out, and I think this can be resolved by doing some simple check against the operator and operands, but what about this case?

int tainted1;
int tainted_cont1 = tainted1;
int tainted_cont2 = tainted1;
int wrongly_tainted = tainted_cont1 ^ tainted_cont2;

As shown above, the operands of xor seems to be different but actually they are same, and this is hard to identify using solely dynamic taint analysis unless we do static control flow analysis.

What if we check against value of the operands? For example, as long as the result of the operation is 0, we mark the result as untainted. This turns out to be a terrible idea:

size_t tainted;
//...
size_t still_tainted = tainted + 0xdeadbeef;
size_t tainted_wrong = still_tainted - tainted;

This should return 0xdeadbeef including the case where the overflow occurs and should have been marked as untainted. But since the result is not 0, it will still be marked as tainted.

And there is also cases where the 2 input values are not necessarily always identical but we have them equal accidentally in some cases when this binary operation is executed. In this case we should not have simply classified it as untainted but the algorithm will. But, if we use forward symbolic execution, this problem might be solved by evaluating and simplifying the equation.

There are also bit-wise operation that may make the senario more complex:

uint32_t tainted;
tainted <<= 0x10;
uint32_t wrongly_tainted = (uint16_t)tainted;

In this case, wrongly_tainted will be marked as tainted, but it should always be 0 since everything has been shift to high 16 bits and low 16 bits are always 0.

The way to solve this is to use bit-wise taint analysis to track the taint flow, and indeed many frameworks have this functionality.

0x03 T-TCOND and T-FCOND

This is hardest problem in taint analysis and symbolic execution, in which people have investigated much time to study. In this paper, for these conditional jump, only the taint status of destination address is considered and abort failure when it is tainted(that means the control flow might be hijacked by attacker). However, the taint status of condition expression is not considered, this will cause problem.

bool tainted;
int wrongly_untained;
if (tainted)
    wrongly_untained = 0xdeadbeef;
else
    wrongly_untained = 0xcafebabe;

As shown, it is obvious that wrongly_untained depends on boolean variable tainted, but it will simply be marked as untainted according to the semantics. The way to solve this is to use static analysis about control flow dependency, but this is hard and not neccessarily accurate especially when the comlexity of the program arises.