rems-project/sail

Why mapping doesn't take function

Closed this issue · 2 comments

I want to implement cm.push, but I've encountered a challenge: using mapping to assemble this instruction would take hundreds of lines. Any suggestions?

mapping clause assembly = CM_PUSH(rlist, spimm) if (sizeof(xlen) == 32) <->
  "cm.push" ^ spc() ^ zcmp_assembly_mapping_32(rlist, spimm) if (sizeof(xlen) == 32)
mapping zcmp_assembly_mapping_32 : (bits(4), bits(2)) <-> string = {
  (0x4, i) <-> "{ra}, " ^ "-" ^ BitStr(to_bits(xlen, i << 5)),
  (0x4, 0b01) <-> "{ra}, -32",
  (0x4, 0b10) <-> "{ra}, -48",
  (0x4, 0b11) <-> "{ra}, -64",
  (0x5, 0b00) <-> "{ra, s0}, -16",
  (0x5, 0b01) <-> "{ra, s0}, -32",
  (0x5, 0b10) <-> "{ra, s0}, -48",
  (0x5, 0b11) <-> "{ra, s0}, -64",
  (0x6, 0b00) <-> "{ra, s0-s1}, -16",
  (0x6, 0b01) <-> "{ra, s0-s1}, -32",
  (0x6, 0b10) <-> "{ra, s0-s1}, -48",
  (0x6, 0b11) <-> "{ra, s0-s1}, -64",
  (0x7, 0b00) <-> "{ra, s0-s2}, -16",
  (0x7, 0b01) <-> "{ra, s0-s2}, -32",
  (0x7, 0b10) <-> "{ra, s0-s2}, -48",
  (0x7, 0b11) <-> "{ra, s0-s2}, -64",
  (0x8, 0b00) <-> "{ra, s0-s3}, -32",
  (0x8, 0b01) <-> "{ra, s0-s3}, -48",
  (0x8, 0b10) <-> "{ra, s0-s3}, -64",
  (0x8, 0b11) <-> "{ra, s0-s3}, -80",
  (0x9, 0b00) <-> "{ra, s0-s4}, -32",
  (0x9, 0b01) <-> "{ra, s0-s4}, -48",
  (0x9, 0b10) <-> "{ra, s0-s4}, -64",
  (0x9, 0b11) <-> "{ra, s0-s4}, -80",
  (0xa, 0b00) <-> "{ra, s0-s5}, -32",
  (0xa, 0b01) <-> "{ra, s0-s5}, -48",
  (0xa, 0b10) <-> "{ra, s0-s5}, -64",
  (0xa, 0b11) <-> "{ra, s0-s5}, -80",
  (0xb, 0b00) <-> "{ra, s0-s6}, -32",
  (0xb, 0b01) <-> "{ra, s0-s6}, -48",
  (0xb, 0b10) <-> "{ra, s0-s6}, -64",
  (0xb, 0b11) <-> "{ra, s0-s6}, -80",
  (0xc, 0b00) <-> "{ra, s0-s7}, -48",
  (0xc, 0b01) <-> "{ra, s0-s7}, -64",
  (0xc, 0b10) <-> "{ra, s0-s7}, -80",
  (0xc, 0b11) <-> "{ra, s0-s7}, -96",
  (0xd, 0b00) <-> "{ra, s0-s8}, -48",
  (0xd, 0b01) <-> "{ra, s0-s8}, -64",
  (0xd, 0b10) <-> "{ra, s0-s8}, -80",
  (0xd, 0b11) <-> "{ra, s0-s8}, -96",
  (0xe, 0b00) <-> "{ra, s0-s9}, -48",
  (0xe, 0b01) <-> "{ra, s0-s9}, -64",
  (0xe, 0b10) <-> "{ra, s0-s9}, -80",
  (0xe, 0b11) <-> "{ra, s0-s9}, -96",
  (0xf, 0b00) <-> "{ra, s0-11}, -64",
  (0xf, 0b01) <-> "{ra, s0-11}, -80",
  (0xf, 0b10) <-> "{ra, s0-11}, -96",
  (0xf, 0b11) <-> "{ra, s0-11}, -112",
}

Mappings can only call other mappings because they have to be reversible and functions don't necessarily have an inverse. Most of the other assembly mappings use, for example, hex_bits_signed_5 for immediate integers. Could you try that? I'm not sure if there are decimal equivalents.

In this case I think you could have a mapping:

mapping zcmp_spimm_assembly_4_7_32 : bits(2) <-> string = {
  0b00 <-> "-16",
  0b01 <-> "-32",
  0b10 <-> "-48",
  0b11 <-> "-64",
}

and call this from the assembly mapping for rlist values 5-7 (not sure about 4?). You could define others for the other spimm mappings for 8-b, c-e and f? This would help a bit but it's still a bit verbose. You might be able to get more clever with some arithmetic.