haybale 0.7.2

Symbolic execution of LLVM IR, written in Rust
Documentation
#include <stdint.h>

struct OneInt {
  int el1;
};

struct TwoInts {
  int el1;
  int el2;
};

struct ThreeInts {
  int el1;
  int el2;
  int el3;
};

struct Mismatched {
  uint8_t el1;
  uint32_t el2;
  uint8_t el3;
};

struct Nested {
  struct TwoInts ti;
  struct Mismatched mm;
};

struct WithArray {
  struct Mismatched mm;
  int arr[10];
  struct Mismatched mm2;
};

// read and write from OneInt
int one_int(int x) {
  volatile struct OneInt oi = { 0 };
  oi.el1 = x;
  return oi.el1 - 3;
}

// read and write from first field in TwoInts
int two_ints_first(int x) {
  volatile struct TwoInts ti = { 0 };
  ti.el1 = x;
  return ti.el1 - 3;
}

// read and write from second field in TwoInts
int two_ints_second(int x) {
  volatile struct TwoInts ti = { 0 };
  ti.el2 = x;
  return ti.el2 - 3;
}

// read and write from both TwoInts fields without getting them confused
int two_ints_both(int x) {
  volatile struct TwoInts ti = { 0 };
  ti.el1 = x + 2;
  ti.el2 = x + 3;
  ti.el1 = ti.el2 - 10;
  ti.el2 = ti.el1 + 7;
  return ti.el2 - 3;
}

// read and write from all fields in ThreeInts without getting them confused
int three_ints(int x, int y) {
  volatile struct ThreeInts ti = { 0 };
  ti.el1 = x + y;
  ti.el2 = x - y;
  ti.el3 = ti.el1 + ti.el2;
  ti.el2 = ti.el3 - 2 * ti.el1;
  ti.el1 = ti.el3 - x;
  return ti.el1 - 3;
}

// ensure that zero-initializing a struct works properly
int zero_initialize(int x) {
  volatile struct ThreeInts ti = { 0 };
  int a = ti.el1 + 2;
  int b = ti.el2 + 4;
  int c = ti.el3 + 6;
  ti.el2 = a + b + c;
  return x - ti.el2;
}

// ensure that non-zero-initializing a struct works properly
int nonzero_initialize(int x) {
  volatile struct ThreeInts ti = { 1, 3, 87 };
  int a = ti.el1 + 2;
  int b = ti.el2 + 4;
  int c = ti.el3 + 6;
  ti.el2 = a + b + c;
  return x - ti.el2;
}

// read and write from the first field in Mismatched
uint8_t mismatched_first(uint8_t x) {
  volatile struct Mismatched mm = { 0 };
  mm.el1 = x;
  return mm.el1 - 3;
}

// read and write from the second field in Mismatched
int mismatched_second(int x) {
  volatile struct Mismatched mm = { 0 };
  mm.el2 = x;
  return mm.el2 - 3;
}

// read and write from the third field in Mismatched
uint8_t mismatched_third(uint8_t x) {
  volatile struct Mismatched mm = { 0 };
  mm.el3 = x;
  return mm.el3 - 3;
}

// read and write from all fields in Mismatched without getting them confused
int mismatched_all(uint8_t x, int y) {
  volatile struct Mismatched mm = { 0 };
  mm.el1 = x + 3;
  mm.el2 = y - 3;
  mm.el3 = mm.el1 - x;
  mm.el1 = mm.el3 - x;
  mm.el2 = mm.el2 + 4;
  mm.el1 = mm.el1 - x;
  mm.el3 = mm.el3 - 5;
  mm.el2 = mm.el2 + y;
  return mm.el1 + mm.el2 + mm.el3;
}

// read and write from the first struct in Nested
int nested_first(int x) {
  volatile struct Nested n = { 0 };
  n.ti.el1 = x;
  n.ti.el2 = 3;
  return n.ti.el1 - n.ti.el2;
}

// read and write from the second struct in Nested
int nested_second(int x) {
  volatile struct Nested n = { 0 };
  n.mm.el2 = x;
  return n.mm.el2 - 3;
}

// read and write from all fields in Nested without getting them confused
int nested_all(uint8_t x, int y) {
  volatile struct Nested n = { 0 };
  n.ti.el2 = y + 3;
  n.mm.el1 = x - 4;
  n.ti.el1 = n.mm.el2 + y;
  n.mm.el3 = n.mm.el1 + 10;
  n.mm.el2 = n.mm.el3 + n.mm.el1;
  n.ti.el2 = n.mm.el3 + n.ti.el1;
  return n.ti.el2 - y;
}

// read and write from the array field in WithArray
int with_array(int x) {
  volatile struct WithArray wa = { 0 };
  wa.arr[4] = x;
  wa.arr[7] = 3;
  return wa.arr[4] - wa.arr[7];
}

// read and write from all fields in WithArray without getting them confused
int with_array_all(int x) {
  volatile struct WithArray wa = { 0 };
  wa.arr[2] = x - 4;
  wa.arr[4] = wa.arr[5] - 3;
  wa.mm.el2 = wa.arr[2];
  wa.mm2.el2 = wa.arr[2] + x + 1;
  return wa.arr[4] + wa.mm2.el2;
}

// manipulate a struct through a pointer
int structptr(int x) {
  volatile struct TwoInts _ti = { 0 };
  volatile struct TwoInts* ti = &_ti;
  ti->el2 = x - 6;
  ti->el1 = ti->el2 + x;
  ti->el2 = 100;
  return ti->el1;
}

// pointer to a particular element of a struct
int structelptr(int x) {
  volatile struct ThreeInts _ti = { 0 };
  volatile struct ThreeInts* ti = &_ti;
  volatile int* iptr = &ti->el2;
  *iptr = 3;
  *iptr = x - *iptr;
  return *iptr;
}

// change the target of a pointer
int changeptr(int x) {
  volatile struct ThreeInts _ti1 = { 0 };
  volatile struct ThreeInts _ti2 = { 0 };
  volatile struct ThreeInts* volatile ti = &_ti1;
  ti->el2 = 7;
  ti = &_ti2;
  ti->el2 = x - 3 - ti->el2;
  ti = &_ti1;
  ti->el2 = 100;
  return _ti2.el2;
}