aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorAryadev Chavali <aryadev@aryadevchavali.com>2024-04-14 03:54:54 +0630
committerAryadev Chavali <aryadev@aryadevchavali.com>2024-04-14 03:54:54 +0630
commit3b912495dec2868f1afae0319471d5ea9451c371 (patch)
tree65d31c9d2b99f2409e90a54dbdd69c588c96a79b
parente12f36466934dcbc48c0d5847591365191504270 (diff)
downloadovm-3b912495dec2868f1afae0319471d5ea9451c371.tar.gz
ovm-3b912495dec2868f1afae0319471d5ea9451c371.tar.bz2
ovm-3b912495dec2868f1afae0319471d5ea9451c371.zip
Created custom functions to convert (h)words to and from bytecode format
Instead of using endian.h that is not portable AND doesn't work with C++, I'll just write my own using a forced union based type punning trick. I've decided to use little endian for the format as well: it seems to be used by most desktop computers so it should make these functions faster to run for most CPUs.
-rw-r--r--lib/base.c72
-rw-r--r--lib/base.h19
-rw-r--r--lib/inst.c4
3 files changed, 87 insertions, 8 deletions
diff --git a/lib/base.c b/lib/base.c
index a2763c4..caa76ae 100644
--- a/lib/base.c
+++ b/lib/base.c
@@ -14,23 +14,87 @@
#include <string.h>
+union hword_pun
+{
+ hword h;
+ byte bytes[HWORD_SIZE];
+};
+
+union word_pun
+{
+ word h;
+ byte bytes[WORD_SIZE];
+};
+
+hword hword_htobc(hword w)
+{
+#if __LITTLE_ENDIAN__
+ return w;
+#else
+ union hword_pun x = {w};
+ union hword_pun y = {0};
+ for (size_t i = 0, j = HWORD_SIZE; i < HWORD_SIZE; ++i, --j)
+ y.bytes[j - 1] = x.bytes[i];
+ return y.h;
+#endif
+}
+
+hword hword_bctoh(hword w)
+{
+#if __LITTLE_ENDIAN__
+ return w;
+#else
+ union hword_pun x = {w};
+ union hword_pun y = {0};
+ for (size_t i = 0, j = HWORD_SIZE; i < HWORD_SIZE; ++i, --j)
+ y.bytes[j - 1] = x.bytes[i];
+ return y.h;
+#endif
+}
+
+word word_htobc(word w)
+{
+#if __LITTLE_ENDIAN__
+ return w;
+#else
+ union word_pun x = {w};
+ union word_pun y = {0};
+ for (size_t i = 0, j = WORD_SIZE; i < WORD_SIZE; ++i, --j)
+ y.bytes[j - 1] = x.bytes[i];
+ return y.h;
+#endif
+}
+
+word word_bctoh(word w)
+{
+#if __LITTLE_ENDIAN__
+ return w;
+#else
+ union word_pun x = {w};
+ union word_pun y = {0};
+ for (size_t i = 0, j = WORD_SIZE; i < WORD_SIZE; ++i, --j)
+ y.bytes[j - 1] = x.bytes[i];
+ return y.h;
+#endif
+}
+
hword convert_bytes_to_hword(byte *bytes)
{
hword be_h = 0;
memcpy(&be_h, bytes, HWORD_SIZE);
- hword h = be32toh(be_h);
+ hword h = hword_bctoh(be_h);
return h;
}
void convert_hword_to_bytes(hword w, byte *bytes)
{
- hword be_h = htobe32(w);
+ hword be_h = hword_htobc(w);
memcpy(bytes, &be_h, HWORD_SIZE);
}
void convert_word_to_bytes(word w, byte *bytes)
{
- word be_w = htobe64(w);
+ word be_w = word_htobc(w);
memcpy(bytes, &be_w, WORD_SIZE);
}
@@ -38,6 +102,6 @@ word convert_bytes_to_word(byte *bytes)
{
word be_w = 0;
memcpy(&be_w, bytes, WORD_SIZE);
- word w = be64toh(be_w);
+ word w = word_bctoh(be_w);
return w;
}
diff --git a/lib/base.h b/lib/base.h
index daa6c58..8ce3510 100644
--- a/lib/base.h
+++ b/lib/base.h
@@ -13,8 +13,6 @@
#ifndef BASE_H
#define BASE_H
-#define _DEFAULT_SOURCE
-#include <endian.h>
#include <stdint.h>
/* Basic macros for a variety of uses. Quite self explanatory. */
@@ -130,4 +128,21 @@ word convert_bytes_to_word(byte *);
*/
void convert_word_to_bytes(word w, byte *buffer);
+/** Convert a half word into bytecode format (little endian)
+ */
+hword hword_htobc(hword);
+
+/** Convert a half word in bytecode format (little endian) to host
+ * format
+ */
+hword hword_bctoh(hword);
+
+/** Convert a word into bytecode format (little endian)
+ */
+word word_htobc(word);
+
+/** Convert a word in bytecode format (little endian) to host format
+ */
+word word_bctoh(word);
+
#endif
diff --git a/lib/inst.c b/lib/inst.c
index 4f013a1..644517e 100644
--- a/lib/inst.c
+++ b/lib/inst.c
@@ -438,7 +438,7 @@ void insts_write_bytecode_file(inst_t *instructions, size_t size, FILE *fp)
void prog_header_write_bytecode(prog_header_t header, darr_t *buffer)
{
- word start = htobe64(header.start_address);
+ word start = word_htobc(header.start_address);
darr_append_bytes(buffer, (byte *)&start, sizeof(start));
}
@@ -447,7 +447,7 @@ void prog_write_bytecode(prog_t *program, darr_t *buffer)
// Write program header
prog_header_write_bytecode(program->header, buffer);
// Write instruction count
- word pcount = htobe64(program->count);
+ word pcount = word_htobc(program->count);
darr_append_bytes(buffer, (byte *)&pcount, sizeof(pcount));
// Write instructions
insts_write_bytecode(program->instructions, program->count, buffer);