mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
Added getStackValue.
This commit is contained in:
parent
532a608e7a
commit
e75732ed9d
@ -20,9 +20,8 @@
|
|||||||
"additionalProperties": false,
|
"additionalProperties": false,
|
||||||
"properties": {
|
"properties": {
|
||||||
"code": {
|
"code": {
|
||||||
"type": "integer",
|
"type": "string",
|
||||||
"minimum": 0,
|
"pattern": "^0x(f8|f9)?[0-9a-f]{2}$",
|
||||||
"maximum": 63999,
|
|
||||||
"description": "Unique byte representation of the opcode."
|
"description": "Unique byte representation of the opcode."
|
||||||
},
|
},
|
||||||
"mnemonic": {
|
"mnemonic": {
|
||||||
|
@ -135,63 +135,63 @@ private fun createBasicBlocks(cfg: ControlFlowGraphBuilder, segment: Instruction
|
|||||||
var branchType: BranchType
|
var branchType: BranchType
|
||||||
var branchLabels: List<Int>
|
var branchLabels: List<Int>
|
||||||
|
|
||||||
when (inst.opcode) {
|
when (inst.opcode.code) {
|
||||||
// Return.
|
// Return.
|
||||||
OP_RET -> {
|
OP_RET.code -> {
|
||||||
branchType = BranchType.Return
|
branchType = BranchType.Return
|
||||||
branchLabels = emptyList()
|
branchLabels = emptyList()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Unconditional jump.
|
// Unconditional jump.
|
||||||
OP_JMP -> {
|
OP_JMP.code -> {
|
||||||
branchType = BranchType.Jump
|
branchType = BranchType.Jump
|
||||||
branchLabels = listOf(inst.args[0].value as Int)
|
branchLabels = listOf(inst.args[0].value as Int)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Conditional jumps.
|
// Conditional jumps.
|
||||||
OP_JMP_ON,
|
OP_JMP_ON.code,
|
||||||
OP_JMP_OFF,
|
OP_JMP_OFF.code,
|
||||||
-> {
|
-> {
|
||||||
branchType = BranchType.ConditionalJump
|
branchType = BranchType.ConditionalJump
|
||||||
branchLabels = listOf(inst.args[0].value as Int)
|
branchLabels = listOf(inst.args[0].value as Int)
|
||||||
}
|
}
|
||||||
OP_JMP_E,
|
OP_JMP_E.code,
|
||||||
OP_JMPI_E,
|
OP_JMPI_E.code,
|
||||||
OP_JMP_NE,
|
OP_JMP_NE.code,
|
||||||
OP_JMPI_NE,
|
OP_JMPI_NE.code,
|
||||||
OP_UJMP_G,
|
OP_UJMP_G.code,
|
||||||
OP_UJMPI_G,
|
OP_UJMPI_G.code,
|
||||||
OP_JMP_G,
|
OP_JMP_G.code,
|
||||||
OP_JMPI_G,
|
OP_JMPI_G.code,
|
||||||
OP_UJMP_L,
|
OP_UJMP_L.code,
|
||||||
OP_UJMPI_L,
|
OP_UJMPI_L.code,
|
||||||
OP_JMP_L,
|
OP_JMP_L.code,
|
||||||
OP_JMPI_L,
|
OP_JMPI_L.code,
|
||||||
OP_UJMP_GE,
|
OP_UJMP_GE.code,
|
||||||
OP_UJMPI_GE,
|
OP_UJMPI_GE.code,
|
||||||
OP_JMP_GE,
|
OP_JMP_GE.code,
|
||||||
OP_JMPI_GE,
|
OP_JMPI_GE.code,
|
||||||
OP_UJMP_LE,
|
OP_UJMP_LE.code,
|
||||||
OP_UJMPI_LE,
|
OP_UJMPI_LE.code,
|
||||||
OP_JMP_LE,
|
OP_JMP_LE.code,
|
||||||
OP_JMPI_LE,
|
OP_JMPI_LE.code,
|
||||||
-> {
|
-> {
|
||||||
branchType = BranchType.ConditionalJump
|
branchType = BranchType.ConditionalJump
|
||||||
branchLabels = listOf(inst.args[2].value as Int)
|
branchLabels = listOf(inst.args[2].value as Int)
|
||||||
}
|
}
|
||||||
OP_SWITCH_JMP -> {
|
OP_SWITCH_JMP.code -> {
|
||||||
branchType = BranchType.ConditionalJump
|
branchType = BranchType.ConditionalJump
|
||||||
branchLabels = inst.args.drop(1).map { it.value as Int }
|
branchLabels = inst.args.drop(1).map { it.value as Int }
|
||||||
}
|
}
|
||||||
|
|
||||||
// Calls.
|
// Calls.
|
||||||
OP_CALL,
|
OP_CALL.code,
|
||||||
OP_VA_CALL,
|
OP_VA_CALL.code,
|
||||||
-> {
|
-> {
|
||||||
branchType = BranchType.Call
|
branchType = BranchType.Call
|
||||||
branchLabels = listOf(inst.args[0].value as Int)
|
branchLabels = listOf(inst.args[0].value as Int)
|
||||||
}
|
}
|
||||||
OP_SWITCH_CALL -> {
|
OP_SWITCH_CALL.code -> {
|
||||||
branchType = BranchType.Call
|
branchType = BranchType.Call
|
||||||
branchLabels = inst.args.drop(1).map { it.value as Int }
|
branchLabels = inst.args.drop(1).map { it.value as Int }
|
||||||
}
|
}
|
||||||
|
@ -16,9 +16,9 @@ fun getMapDesignations(
|
|||||||
var cfg: ControlFlowGraph? = null
|
var cfg: ControlFlowGraph? = null
|
||||||
|
|
||||||
for (inst in func0Segment.instructions) {
|
for (inst in func0Segment.instructions) {
|
||||||
when (inst.opcode) {
|
when (inst.opcode.code) {
|
||||||
OP_MAP_DESIGNATE,
|
OP_MAP_DESIGNATE.code,
|
||||||
OP_MAP_DESIGNATE_EX,
|
OP_MAP_DESIGNATE_EX.code,
|
||||||
-> {
|
-> {
|
||||||
if (cfg == null) {
|
if (cfg == null) {
|
||||||
cfg = ControlFlowGraph.create(instructionSegments)
|
cfg = ControlFlowGraph.create(instructionSegments)
|
||||||
@ -47,7 +47,7 @@ fun getMapDesignations(
|
|||||||
mapDesignations[areaId[0]!!] = variantId[0]!!
|
mapDesignations[areaId[0]!!] = variantId[0]!!
|
||||||
}
|
}
|
||||||
|
|
||||||
OP_BB_MAP_DESIGNATE -> {
|
OP_BB_MAP_DESIGNATE.code -> {
|
||||||
mapDesignations[inst.args[0].value as Int] = inst.args[2].value as Int
|
mapDesignations[inst.args[0].value as Int] = inst.args[2].value as Int
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -7,17 +7,13 @@ import kotlin.math.min
|
|||||||
|
|
||||||
private val logger = KotlinLogging.logger {}
|
private val logger = KotlinLogging.logger {}
|
||||||
|
|
||||||
const val MIN_REGISTER_VALUE = MIN_SIGNED_DWORD_VALUE
|
|
||||||
const val MAX_REGISTER_VALUE = MAX_SIGNED_DWORD_VALUE
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Computes the possible values of a register right before a specific instruction.
|
* Computes the possible values of a register right before a specific instruction.
|
||||||
*/
|
*/
|
||||||
fun getRegisterValue(cfg: ControlFlowGraph, instruction: Instruction, register: Int): ValueSet {
|
fun getRegisterValue(cfg: ControlFlowGraph, instruction: Instruction, register: Int): ValueSet {
|
||||||
val block = cfg.getBlockForInstruction(instruction)
|
val block = cfg.getBlockForInstruction(instruction)
|
||||||
|
|
||||||
return findValues(
|
return RegisterValueFinder().find(
|
||||||
Context(),
|
|
||||||
mutableSetOf(),
|
mutableSetOf(),
|
||||||
block,
|
block,
|
||||||
block.indexOfInstruction(instruction),
|
block.indexOfInstruction(instruction),
|
||||||
@ -25,58 +21,56 @@ fun getRegisterValue(cfg: ControlFlowGraph, instruction: Instruction, register:
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
private class Context {
|
private class RegisterValueFinder {
|
||||||
var iterations = 0
|
var iterations = 0
|
||||||
}
|
|
||||||
|
|
||||||
private fun findValues(
|
fun find(
|
||||||
ctx: Context,
|
|
||||||
path: MutableSet<BasicBlock>,
|
path: MutableSet<BasicBlock>,
|
||||||
block: BasicBlock,
|
block: BasicBlock,
|
||||||
end: Int,
|
end: Int,
|
||||||
register: Int,
|
register: Int,
|
||||||
): ValueSet {
|
): ValueSet {
|
||||||
if (++ctx.iterations > 100) {
|
if (++iterations > 100) {
|
||||||
logger.warn { "Too many iterations." }
|
logger.warn { "Too many iterations." }
|
||||||
return ValueSet.ofInterval(MIN_REGISTER_VALUE, MAX_REGISTER_VALUE)
|
return ValueSet.all()
|
||||||
}
|
}
|
||||||
|
|
||||||
for (i in end - 1 downTo block.start) {
|
for (i in end - 1 downTo block.start) {
|
||||||
val instruction = block.segment.instructions[i]
|
val instruction = block.segment.instructions[i]
|
||||||
val args = instruction.args
|
val args = instruction.args
|
||||||
|
|
||||||
when (instruction.opcode) {
|
when (instruction.opcode.code) {
|
||||||
OP_LET -> {
|
OP_LET.code -> {
|
||||||
if (args[0].value == register) {
|
if (args[0].value == register) {
|
||||||
return findValues(ctx, LinkedHashSet(path), block, i, args[1].value as Int)
|
return find(LinkedHashSet(path), block, i, args[1].value as Int)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
OP_LETI,
|
OP_LETI.code,
|
||||||
OP_LETB,
|
OP_LETB.code,
|
||||||
OP_LETW,
|
OP_LETW.code,
|
||||||
OP_SYNC_LETI,
|
OP_SYNC_LETI.code,
|
||||||
-> {
|
-> {
|
||||||
if (args[0].value == register) {
|
if (args[0].value == register) {
|
||||||
return ValueSet.of(args[1].value as Int)
|
return ValueSet.of(args[1].value as Int)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
OP_SET -> {
|
OP_SET.code -> {
|
||||||
if (args[0].value == register) {
|
if (args[0].value == register) {
|
||||||
return ValueSet.of(1)
|
return ValueSet.of(1)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
OP_CLEAR -> {
|
OP_CLEAR.code -> {
|
||||||
if (args[0].value == register) {
|
if (args[0].value == register) {
|
||||||
return ValueSet.of(0)
|
return ValueSet.of(0)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
OP_REV -> {
|
OP_REV.code -> {
|
||||||
if (args[0].value == register) {
|
if (args[0].value == register) {
|
||||||
val prevVals = findValues(ctx, LinkedHashSet(path), block, i, register)
|
val prevVals = find(LinkedHashSet(path), block, i, register)
|
||||||
|
|
||||||
return if (prevVals.size == 1L && prevVals[0] == 0) {
|
return if (prevVals.size == 1L && prevVals[0] == 0) {
|
||||||
ValueSet.of(1)
|
ValueSet.of(1)
|
||||||
@ -88,61 +82,60 @@ private fun findValues(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
OP_ADDI -> {
|
OP_ADDI.code -> {
|
||||||
if (args[0].value == register) {
|
if (args[0].value == register) {
|
||||||
val prevVals = findValues(ctx, LinkedHashSet(path), block, i, register)
|
val prevVals = find(LinkedHashSet(path), block, i, register)
|
||||||
prevVals += args[1].value as Int
|
prevVals += args[1].value as Int
|
||||||
return prevVals
|
return prevVals
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
OP_SUBI -> {
|
OP_SUBI.code -> {
|
||||||
if (args[0].value == register) {
|
if (args[0].value == register) {
|
||||||
val prevVals = findValues(ctx, LinkedHashSet(path), block, i, register)
|
val prevVals = find(LinkedHashSet(path), block, i, register)
|
||||||
prevVals -= args[1].value as Int
|
prevVals -= args[1].value as Int
|
||||||
return prevVals
|
return prevVals
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
OP_MULI -> {
|
OP_MULI.code -> {
|
||||||
if (args[0].value == register) {
|
if (args[0].value == register) {
|
||||||
val prevVals = findValues(ctx, LinkedHashSet(path), block, i, register)
|
val prevVals = find(LinkedHashSet(path), block, i, register)
|
||||||
prevVals *= args[1].value as Int
|
prevVals *= args[1].value as Int
|
||||||
return prevVals
|
return prevVals
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
OP_DIVI -> {
|
OP_DIVI.code -> {
|
||||||
if (args[0].value == register) {
|
if (args[0].value == register) {
|
||||||
val prevVals = findValues(ctx, LinkedHashSet(path), block, i, register)
|
val prevVals = find(LinkedHashSet(path), block, i, register)
|
||||||
prevVals /= args[1].value as Int
|
prevVals /= args[1].value as Int
|
||||||
return prevVals
|
return prevVals
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
OP_IF_ZONE_CLEAR -> {
|
OP_IF_ZONE_CLEAR.code -> {
|
||||||
if (args[0].value == register) {
|
if (args[0].value == register) {
|
||||||
return ValueSet.ofInterval(0, 1)
|
return ValueSet.ofInterval(0, 1)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
OP_GET_DIFFLVL -> {
|
OP_GET_DIFFLVL.code -> {
|
||||||
if (args[0].value == register) {
|
if (args[0].value == register) {
|
||||||
return ValueSet.ofInterval(0, 2)
|
return ValueSet.ofInterval(0, 2)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
OP_GET_SLOTNUMBER -> {
|
OP_GET_SLOTNUMBER.code -> {
|
||||||
if (args[0].value == register) {
|
if (args[0].value == register) {
|
||||||
return ValueSet.ofInterval(0, 3)
|
return ValueSet.ofInterval(0, 3)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
OP_GET_RANDOM -> {
|
OP_GET_RANDOM.code -> {
|
||||||
if (args[1].value == register) {
|
if (args[1].value == register) {
|
||||||
// TODO: undefined values.
|
// TODO: undefined values.
|
||||||
val min = findValues(
|
val min = find(
|
||||||
ctx,
|
|
||||||
LinkedHashSet(path),
|
LinkedHashSet(path),
|
||||||
block,
|
block,
|
||||||
i,
|
i,
|
||||||
@ -150,8 +143,7 @@ private fun findValues(
|
|||||||
).minOrNull()!!
|
).minOrNull()!!
|
||||||
|
|
||||||
val max = max(
|
val max = max(
|
||||||
findValues(
|
find(
|
||||||
ctx,
|
|
||||||
LinkedHashSet(path),
|
LinkedHashSet(path),
|
||||||
block,
|
block,
|
||||||
i,
|
i,
|
||||||
@ -164,14 +156,14 @@ private fun findValues(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
OP_STACK_PUSHM,
|
OP_STACK_PUSHM.code,
|
||||||
OP_STACK_POPM,
|
OP_STACK_POPM.code,
|
||||||
-> {
|
-> {
|
||||||
val minReg = args[0].value as Int
|
val minReg = args[0].value as Int
|
||||||
val maxReg = args[0].value as Int + args[1].value as Int
|
val maxReg = args[0].value as Int + args[1].value as Int
|
||||||
|
|
||||||
if (register in minReg until maxReg) {
|
if (register in minReg until maxReg) {
|
||||||
return ValueSet.ofInterval(MIN_REGISTER_VALUE, MAX_REGISTER_VALUE)
|
return ValueSet.all()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -191,10 +183,7 @@ private fun findValues(
|
|||||||
reg_param.access == ParamAccess.ReadWrite) &&
|
reg_param.access == ParamAccess.ReadWrite) &&
|
||||||
regRef + k == register
|
regRef + k == register
|
||||||
) {
|
) {
|
||||||
return ValueSet.ofInterval(
|
return ValueSet.all()
|
||||||
MIN_REGISTER_VALUE,
|
|
||||||
MAX_REGISTER_VALUE,
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -209,11 +198,10 @@ private fun findValues(
|
|||||||
for (from in block.from) {
|
for (from in block.from) {
|
||||||
// Bail out from loops.
|
// Bail out from loops.
|
||||||
if (from in path) {
|
if (from in path) {
|
||||||
values.setInterval(MIN_REGISTER_VALUE, MAX_REGISTER_VALUE)
|
return ValueSet.all()
|
||||||
break
|
|
||||||
}
|
}
|
||||||
|
|
||||||
values.union(findValues(ctx, LinkedHashSet(path), from, from.end, register))
|
values.union(find(LinkedHashSet(path), from, from.end, register))
|
||||||
}
|
}
|
||||||
|
|
||||||
// If values is empty at this point, we know nothing ever sets the register's value and it still
|
// If values is empty at this point, we know nothing ever sets the register's value and it still
|
||||||
@ -223,4 +211,5 @@ private fun findValues(
|
|||||||
}
|
}
|
||||||
|
|
||||||
return values
|
return values
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,103 @@
|
|||||||
package world.phantasmal.lib.assembly.dataFlowAnalysis
|
package world.phantasmal.lib.assembly.dataFlowAnalysis
|
||||||
|
|
||||||
import world.phantasmal.lib.assembly.Instruction
|
import mu.KotlinLogging
|
||||||
|
import world.phantasmal.lib.assembly.*
|
||||||
|
|
||||||
fun getStackValue(cfg: ControlFlowGraph, instruction: Instruction, position: Int): ValueSet =
|
private val logger = KotlinLogging.logger {}
|
||||||
TODO()
|
|
||||||
|
/**
|
||||||
|
* Computes the possible values of a stack element at the nth position from the top right before a
|
||||||
|
* specific instruction.
|
||||||
|
*/
|
||||||
|
fun getStackValue(cfg: ControlFlowGraph, instruction: Instruction, position: Int): ValueSet {
|
||||||
|
val block = cfg.getBlockForInstruction(instruction)
|
||||||
|
|
||||||
|
return StackValueFinder().find(
|
||||||
|
mutableSetOf(),
|
||||||
|
cfg,
|
||||||
|
block,
|
||||||
|
block.indexOfInstruction(instruction),
|
||||||
|
position,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
private class StackValueFinder {
|
||||||
|
var iterations = 0
|
||||||
|
|
||||||
|
fun find(
|
||||||
|
path: MutableSet<BasicBlock>,
|
||||||
|
cfg: ControlFlowGraph,
|
||||||
|
block: BasicBlock,
|
||||||
|
end: Int,
|
||||||
|
position: Int,
|
||||||
|
): ValueSet {
|
||||||
|
if (++iterations > 100) {
|
||||||
|
logger.warn { "Too many iterations." }
|
||||||
|
return ValueSet.all()
|
||||||
|
}
|
||||||
|
|
||||||
|
var pos = position
|
||||||
|
|
||||||
|
for (i in end - 1 downTo block.start) {
|
||||||
|
val instruction = block.segment.instructions[i]
|
||||||
|
|
||||||
|
if (instruction.opcode.stack == StackInteraction.Pop) {
|
||||||
|
pos += instruction.opcode.params.size
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
val args = instruction.args
|
||||||
|
|
||||||
|
when (instruction.opcode.code) {
|
||||||
|
OP_ARG_PUSHR.code -> {
|
||||||
|
if (pos == 0) {
|
||||||
|
return getRegisterValue(cfg, instruction, args[0].value as Int)
|
||||||
|
} else {
|
||||||
|
pos--
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
OP_ARG_PUSHL.code,
|
||||||
|
OP_ARG_PUSHB.code,
|
||||||
|
OP_ARG_PUSHW.code,
|
||||||
|
-> {
|
||||||
|
if (pos == 0) {
|
||||||
|
return ValueSet.of(args[0].value as Int)
|
||||||
|
} else {
|
||||||
|
pos--
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
OP_ARG_PUSHA.code,
|
||||||
|
OP_ARG_PUSHO.code,
|
||||||
|
OP_ARG_PUSHS.code,
|
||||||
|
-> {
|
||||||
|
if (pos == 0) {
|
||||||
|
return ValueSet.all()
|
||||||
|
} else {
|
||||||
|
pos--
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
else -> break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
val values = ValueSet.empty()
|
||||||
|
path.add(block)
|
||||||
|
|
||||||
|
for (from in block.from) {
|
||||||
|
// Bail out from loops.
|
||||||
|
if (from in path) {
|
||||||
|
return ValueSet.all()
|
||||||
|
}
|
||||||
|
|
||||||
|
values.union(find(LinkedHashSet(path), cfg, from, from.end, pos))
|
||||||
|
}
|
||||||
|
|
||||||
|
return values
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -8,8 +8,7 @@ import kotlin.math.min
|
|||||||
*/
|
*/
|
||||||
class ValueSet private constructor(private val intervals: MutableList<Interval>) : Iterable<Int> {
|
class ValueSet private constructor(private val intervals: MutableList<Interval>) : Iterable<Int> {
|
||||||
val size: Long
|
val size: Long
|
||||||
get() =
|
get() = intervals.fold(0L) { acc, i -> acc + i.end - i.start + 1L }
|
||||||
intervals.fold(0L) { acc, i -> acc + i.end - i.start + 1L }
|
|
||||||
|
|
||||||
operator fun get(i: Int): Int? {
|
operator fun get(i: Int): Int? {
|
||||||
var idx = i
|
var idx = i
|
||||||
@ -162,8 +161,8 @@ class ValueSet private constructor(private val intervals: MutableList<Interval>)
|
|||||||
*/
|
*/
|
||||||
operator fun divAssign(s: Int) {
|
operator fun divAssign(s: Int) {
|
||||||
for (int in intervals) {
|
for (int in intervals) {
|
||||||
int.start = int.start / s
|
int.start /= s
|
||||||
int.end = int.end / s
|
int.end /= s
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -245,6 +244,11 @@ class ValueSet private constructor(private val intervals: MutableList<Interval>)
|
|||||||
*/
|
*/
|
||||||
fun empty(): ValueSet = ValueSet(mutableListOf())
|
fun empty(): ValueSet = ValueSet(mutableListOf())
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a [ValueSet] containing all possible Int values.
|
||||||
|
*/
|
||||||
|
fun all(): ValueSet = ofInterval(Int.MIN_VALUE, Int.MAX_VALUE)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a [ValueSet] with a single initial [value].
|
* Returns a [ValueSet] with a single initial [value].
|
||||||
*/
|
*/
|
||||||
|
@ -405,9 +405,9 @@ private fun parseInstructionsSegment(
|
|||||||
var dropThrough = true
|
var dropThrough = true
|
||||||
|
|
||||||
for (i in instructions.size - 1 downTo 0) {
|
for (i in instructions.size - 1 downTo 0) {
|
||||||
val opcode = instructions[i].opcode
|
val opcode = instructions[i].opcode.code
|
||||||
|
|
||||||
if (opcode == OP_RET || opcode == OP_JMP) {
|
if (opcode == OP_RET.code || opcode == OP_JMP.code) {
|
||||||
dropThrough = false
|
dropThrough = false
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
@ -81,14 +81,14 @@ class GetRegisterValueTests {
|
|||||||
val r0 = getRegisterValue(cfg, im[0].instructions[2], 0)
|
val r0 = getRegisterValue(cfg, im[0].instructions[2], 0)
|
||||||
|
|
||||||
assertEquals(MAX_REGISTER_VALUES_SIZE, r0.size)
|
assertEquals(MAX_REGISTER_VALUES_SIZE, r0.size)
|
||||||
assertEquals(MIN_REGISTER_VALUE, r0.minOrNull())
|
assertEquals(Int.MIN_VALUE, r0.minOrNull())
|
||||||
assertEquals(MAX_REGISTER_VALUE, r0.maxOrNull())
|
assertEquals(Int.MAX_VALUE, r0.maxOrNull())
|
||||||
|
|
||||||
val r1 = getRegisterValue(cfg, im[0].instructions[2], 1)
|
val r1 = getRegisterValue(cfg, im[0].instructions[2], 1)
|
||||||
|
|
||||||
assertEquals(MAX_REGISTER_VALUES_SIZE, r1.size)
|
assertEquals(MAX_REGISTER_VALUES_SIZE, r1.size)
|
||||||
assertEquals(MIN_REGISTER_VALUE, r1.minOrNull())
|
assertEquals(Int.MIN_VALUE, r1.minOrNull())
|
||||||
assertEquals(MAX_REGISTER_VALUE, r1.maxOrNull())
|
assertEquals(Int.MAX_VALUE, r1.maxOrNull())
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
@ -61,7 +61,7 @@ class ValueSetTests {
|
|||||||
fun plusAssign_integer_overflow() {
|
fun plusAssign_integer_overflow() {
|
||||||
// The set of all integers should stay the same after adding any integer.
|
// The set of all integers should stay the same after adding any integer.
|
||||||
for (i in Int.MIN_VALUE..Int.MAX_VALUE step 10_000_000) {
|
for (i in Int.MIN_VALUE..Int.MAX_VALUE step 10_000_000) {
|
||||||
val vs = ValueSet.ofInterval(Int.MIN_VALUE, Int.MAX_VALUE)
|
val vs = ValueSet.all()
|
||||||
vs += i
|
vs += i
|
||||||
|
|
||||||
assertEquals(1L shl 32, vs.size)
|
assertEquals(1L shl 32, vs.size)
|
||||||
@ -100,7 +100,7 @@ class ValueSetTests {
|
|||||||
fun minusAssign_integer_underflow() {
|
fun minusAssign_integer_underflow() {
|
||||||
// The set of all integers should stay the same after subtracting any integer.
|
// The set of all integers should stay the same after subtracting any integer.
|
||||||
for (i in Int.MIN_VALUE..Int.MAX_VALUE step 10_000_000) {
|
for (i in Int.MIN_VALUE..Int.MAX_VALUE step 10_000_000) {
|
||||||
val vs = ValueSet.ofInterval(Int.MIN_VALUE, Int.MAX_VALUE)
|
val vs = ValueSet.all()
|
||||||
vs -= i
|
vs -= i
|
||||||
|
|
||||||
assertEquals(1L shl 32, vs.size)
|
assertEquals(1L shl 32, vs.size)
|
||||||
|
Loading…
Reference in New Issue
Block a user