mirror of
https://github.com/radareorg/radare2.git
synced 2024-12-11 15:04:23 +00:00
9a32f3db86
* io.cache.read|write are now true by default * io.cache-related unit tests disabled for now
427 lines
12 KiB
C
427 lines
12 KiB
C
#include <r_util.h>
|
|
#include <r_io.h>
|
|
#include <stdlib.h>
|
|
#include "minunit.h"
|
|
|
|
bool test_buf(RBuffer *b) {
|
|
ut8 buffer[1024] = {0};
|
|
const char *content = "Something To\nSay Here..";
|
|
const int length = 23;
|
|
int r;
|
|
|
|
ut64 buf_sz = r_buf_size (b);
|
|
mu_assert_eq (buf_sz, length, "file size should be computed");
|
|
|
|
r = r_buf_read (b, buffer, length);
|
|
mu_assert_eq (r, length, "r_buf_read_at failed");
|
|
mu_assert_memeq (buffer, (ut8 *)content, length, "r_buf_read_at has corrupted content");
|
|
|
|
const char *s = "This is a new content";
|
|
const size_t sl = strlen (s);
|
|
bool res = r_buf_set_bytes (b, (ut8 *)s, sl);
|
|
mu_assert ("New content should be written", res);
|
|
|
|
r_buf_seek (b, 0, R_BUF_SET);
|
|
r = r_buf_read (b, buffer, sl);
|
|
mu_assert_eq (r, sl, "r_buf_read_at failed");
|
|
mu_assert_memeq (buffer, (ut8 *)s, sl, "r_buf_read_at has corrupted content");
|
|
|
|
r_buf_seek (b, 0, R_BUF_SET);
|
|
r = r_buf_read (b, buffer, 3);
|
|
mu_assert_eq (r, 3, "r_buf_read_at failed");
|
|
mu_assert_memeq (buffer, (ut8 *)"Thi", 3, "r_buf_read_at has corrupted content");
|
|
r = r_buf_read (b, buffer, 5);
|
|
mu_assert_eq (r, 5, "r_buf_read_at failed");
|
|
mu_assert_memeq (buffer, (ut8 *)"s is ", 5, "r_buf_read_at has corrupted content");
|
|
|
|
const char *s2 = ", hello world";
|
|
const size_t s2l = strlen (s2);
|
|
res = r_buf_append_string (b, s2);
|
|
mu_assert ("string should be appended", res);
|
|
|
|
buf_sz = r_buf_size (b);
|
|
mu_assert_eq (buf_sz, sl + s2l, "file size should be computed");
|
|
|
|
res = r_buf_resize (b, 10);
|
|
mu_assert ("file should be resized", res);
|
|
buf_sz = r_buf_size (b);
|
|
mu_assert_eq (buf_sz, 10, "file size should be 10");
|
|
|
|
const int rl = r_buf_read_at (b, 1, buffer, sizeof (buffer));
|
|
mu_assert_eq (rl, 9, "only 9 bytes can be read from offset 1");
|
|
mu_assert_memeq (buffer, (ut8 *)"his is a ", 9, "read right bytes from offset 1");
|
|
|
|
r_buf_set_bytes (b, (ut8 *)"World", strlen ("World"));
|
|
char *base = r_buf_tostring (b);
|
|
mu_assert_notnull (base, "string should be there");
|
|
mu_assert_streq (base, "World", "World there");
|
|
free (base);
|
|
|
|
const char *s3 = "Hello ";
|
|
res = r_buf_prepend_bytes (b, (const ut8 *)s3, strlen (s3));
|
|
mu_assert ("bytes should be prepended", res);
|
|
char *st = r_buf_tostring (b);
|
|
mu_assert_notnull (st, "string should be there");
|
|
mu_assert_streq (st, "Hello World", "hello world there");
|
|
free (st);
|
|
|
|
r_buf_insert_bytes (b, 5, (ut8 *)",", 1);
|
|
char *st2 = r_buf_tostring (b);
|
|
mu_assert_notnull (st2, "string should be there");
|
|
mu_assert_streq (st2, "Hello, World", "comma inserted");
|
|
free (st2);
|
|
|
|
r = r_buf_seek (b, 0x100, R_BUF_SET);
|
|
mu_assert_eq (r, 0x100, "moving seek out of current length");
|
|
r = r_buf_write (b, (ut8 *)"mydata", 6);
|
|
mu_assert_eq (r, 6, "writes 6 bytes");
|
|
r = r_buf_read_at (b, 0xf0, buffer, sizeof (buffer));
|
|
mu_assert_eq (r, 0x16, "read 16 bytes at the end of gap and new data");
|
|
mu_assert_memeq (buffer, (ut8 *)"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00", 16, "first bytes should be 0");
|
|
mu_assert_memeq (buffer + 0x10, (ut8 *)"mydata", 6, "then there is mydata");
|
|
|
|
r_buf_set_bytes (b, (ut8 *)"Hello", 5);
|
|
RBuffer *sec_buf = r_buf_new_with_bytes ((ut8 *)" second", 7);
|
|
res = r_buf_append_buf (b, sec_buf);
|
|
mu_assert ("append buf should succeed", res);
|
|
char *st3 = r_buf_tostring (b);
|
|
mu_assert_streq (st3, "Hello second", "append buf correctly");
|
|
free (st3);
|
|
r_buf_free (sec_buf);
|
|
|
|
sec_buf = r_buf_new_with_bytes ((ut8 *)"123456789", 9);
|
|
res = r_buf_append_buf_slice (b, sec_buf, 5, 3);
|
|
mu_assert ("append buf slice should succeed", res);
|
|
char *st4 = r_buf_tostring (b);
|
|
mu_assert_streq (st4, "Hello second678", "append buf slice correctly");
|
|
free (st4);
|
|
r_buf_free (sec_buf);
|
|
|
|
return MU_PASSED;
|
|
}
|
|
|
|
bool test_r_buf_file(void) {
|
|
RBuffer *b;
|
|
char *filename = "r2-XXXXXX";
|
|
const char *content = "Something To\nSay Here..";
|
|
const int length = 23;
|
|
|
|
// Prepare file
|
|
int fd = r_file_mkstemp ("", &filename);
|
|
mu_assert_neq ((ut64)fd, (ut64)-1, "mkstemp failed...");
|
|
if (write (fd, content, length) != length) {
|
|
return false;
|
|
}
|
|
close (fd);
|
|
|
|
b = r_buf_new_file (filename, O_RDWR, 0);
|
|
mu_assert_notnull (b, "r_buf_new_file failed");
|
|
|
|
if (test_buf (b) != MU_PASSED) {
|
|
mu_fail ("test failed");
|
|
}
|
|
|
|
// Cleanup
|
|
r_buf_free (b);
|
|
unlink (filename);
|
|
free (filename);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_buf_bytes(void) {
|
|
RBuffer *b;
|
|
const char *content = "Something To\nSay Here..";
|
|
const int length = 23;
|
|
|
|
b = r_buf_new_with_bytes ((const ut8 *)content, length);
|
|
mu_assert_notnull (b, "r_buf_new_with_bytes failed");
|
|
|
|
if (test_buf (b) != MU_PASSED) {
|
|
mu_fail ("test failed");
|
|
}
|
|
|
|
// Cleanup
|
|
r_buf_free (b);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_buf_mmap(void) {
|
|
RBuffer *b;
|
|
char *filename = "r2-XXXXXX";
|
|
const char *content = "Something To\nSay Here..";
|
|
const int length = 23;
|
|
|
|
// Prepare file
|
|
int fd = r_file_mkstemp ("", &filename);
|
|
mu_assert_neq ((long long)fd, -1LL, "mkstemp failed...");
|
|
if (write (fd, content, length) != length) {
|
|
return false;
|
|
}
|
|
close (fd);
|
|
|
|
b = r_buf_new_mmap (filename, R_PERM_RW);
|
|
mu_assert_notnull (b, "r_buf_new_mmap failed");
|
|
|
|
if (test_buf (b) != MU_PASSED) {
|
|
unlink (filename);
|
|
mu_fail ("test failed");
|
|
}
|
|
|
|
// Cleanup
|
|
r_buf_free (b);
|
|
unlink (filename);
|
|
free (filename);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_buf_io(void) {
|
|
const char *content = "Something To\nSay Here..";
|
|
const int length = 23;
|
|
|
|
RIO *io = r_io_new ();
|
|
RIODesc *desc = r_io_open_at (io, "file:///tmp/r-buf-io.test", R_PERM_RW | R_PERM_CREAT, 0644, 0);
|
|
mu_assert_notnull (desc, "file should be opened for writing");
|
|
|
|
bool res = r_io_write_at (io, 0, (ut8 *)content, length);
|
|
mu_assert ("initial content should be written", res);
|
|
|
|
RIOBind bnd;
|
|
r_io_bind (io, &bnd);
|
|
|
|
RBuffer *b = r_buf_new_with_io (&bnd, desc->fd);
|
|
mu_assert_notnull (b, "r_buf_new_file failed");
|
|
|
|
if (test_buf (b) != MU_PASSED) {
|
|
mu_fail ("test failed");
|
|
}
|
|
|
|
// Cleanup
|
|
r_buf_free (b);
|
|
r_io_close (io);
|
|
r_io_free (io);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_buf_sparse(void) {
|
|
RBuffer *b;
|
|
const char *content = "Something To\nSay Here..";
|
|
const int length = 23;
|
|
|
|
b = r_buf_new_sparse (0);
|
|
mu_assert_notnull (b, "r_buf_new_file failed");
|
|
|
|
r_buf_write (b, (ut8 *)content, length);
|
|
r_buf_seek (b, 0, R_BUF_SET);
|
|
|
|
if (test_buf (b) != MU_PASSED) {
|
|
mu_fail ("test failed");
|
|
}
|
|
|
|
// Cleanup
|
|
r_buf_free (b);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_buf_sparse2(void) {
|
|
RBuffer *b = r_buf_new_sparse (0xff);
|
|
r_buf_write (b, (ut8 *)"aaaa", 4);
|
|
r_buf_write (b, (ut8 *)"bbbbb", 5);
|
|
r_buf_write (b, (ut8 *)"cccccc", 6);
|
|
r_buf_write_at (b, 2, (ut8 *)"D", 1);
|
|
r_buf_write_at (b, 7, (ut8 *)"EEE", 3);
|
|
|
|
ut8 tmp[20];
|
|
int r = r_buf_read_at (b, 0, tmp, sizeof (tmp));
|
|
mu_assert_eq (r, 15, "read only 15 bytes");
|
|
mu_assert_memeq (tmp, (ut8 *)"aaDabbbEEEccccc", 15, "read the right bytes");
|
|
|
|
bool res = r_buf_resize (b, 0);
|
|
mu_assert ("resized to 0", res);
|
|
|
|
r = r_buf_read_at (b, 0, tmp, sizeof (tmp));
|
|
mu_assert_eq (r, 0, "nothing to read");
|
|
|
|
r_buf_write_at (b, 3, (ut8 *)"aaaa", 4);
|
|
r = r_buf_read_at (b, 0, tmp, sizeof (tmp));
|
|
mu_assert_eq (r, 7, "read the initial 0xff bytes");
|
|
mu_assert_memeq (tmp, (ut8 *)"\xff\xff\xff\x61\x61\x61\x61", 7, "right 7 bytes");
|
|
|
|
res = r_buf_resize (b, 10);
|
|
mu_assert ("resized to 10", res);
|
|
|
|
ut64 sz = r_buf_size (b);
|
|
mu_assert_eq (sz, 10, "size is 10");
|
|
r = r_buf_read_at (b, 0, tmp, sizeof (tmp));
|
|
mu_assert_eq (r, 10, "read the initial/final 0xff bytes");
|
|
mu_assert_memeq (tmp, (ut8 *)"\xff\xff\xff\x61\x61\x61\x61\xff\xff\xff", 10, "right 10 bytes");
|
|
|
|
r = r_buf_write_at (b, 0x100, (ut8 *)"ABCDEF", 6);
|
|
mu_assert_eq (r, 6, "write 6 bytes at 0x100");
|
|
r = r_buf_read_at (b, 0xfe, tmp, sizeof (tmp));
|
|
mu_assert_eq (r, 8, "read 8 bytes");
|
|
mu_assert_memeq (tmp, (ut8 *)"\xff\xff\x41\x42\x43\x44\x45\x46", 8, "right bytes");
|
|
|
|
sz = r_buf_size (b);
|
|
mu_assert_eq (sz, 0x106, "size is 0x106");
|
|
|
|
r_buf_free (b);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_buf_bytes_steal(void) {
|
|
RBuffer *b;
|
|
const char *content = "Something To\nSay Here..";
|
|
const int length = 23;
|
|
|
|
b = r_buf_new_with_bytes ((const ut8 *)content, length);
|
|
mu_assert_notnull (b, "r_buf_new_file failed");
|
|
char *s = r_buf_tostring (b);
|
|
mu_assert_streq (s, content, "content is right");
|
|
free (s);
|
|
|
|
// Cleanup
|
|
r_buf_free (b);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_buf_format(void) {
|
|
RBuffer *b = r_buf_new ();
|
|
uint16_t a[] = {0xdead, 0xbeef, 0xcafe, 0xbabe};
|
|
ut8 buf[4 * sizeof (uint16_t)];
|
|
|
|
r_buf_fwrite (b, (ut8 *)a, "4s", 1);
|
|
r_buf_read_at (b, 0, buf, sizeof (buf));
|
|
mu_assert_memeq (buf, (ut8 *)"\xad\xde\xef\xbe\xfe\xca\xbe\xba", sizeof(buf), "fwrite");
|
|
|
|
r_buf_fread_at (b, 0, (ut8 *)a, "S", 4);
|
|
mu_assert_eq (a[0], 0xadde, "first");
|
|
mu_assert_eq (a[1], 0xefbe, "second");
|
|
mu_assert_eq (a[2], 0xfeca, "third");
|
|
mu_assert_eq (a[3], 0xbeba, "fourth");
|
|
|
|
r_buf_free (b);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_buf_with_buf(void) {
|
|
const char *content = "Something To\nSay Here..";
|
|
const int length = 23;
|
|
RBuffer *buf = r_buf_new_with_bytes ((ut8 *)content, length);
|
|
|
|
RBuffer *b = r_buf_new_with_buf (buf);
|
|
mu_assert_notnull (b, "r_buf_new_with_buf failed");
|
|
r_buf_free (buf);
|
|
|
|
if (test_buf (b) != MU_PASSED) {
|
|
mu_fail ("r_buf_with_buf failed");
|
|
}
|
|
|
|
// Cleanup
|
|
r_buf_free (b);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_buf_slice(void) {
|
|
const char *content = "AAAAAAAAAASomething To\nSay Here..BBBBBBBBBB";
|
|
const int length = strlen (content);
|
|
RBuffer *buf = r_buf_new_with_bytes ((ut8 *)content, length);
|
|
ut8 buffer[1024];
|
|
|
|
RBuffer *b = r_buf_new_slice (buf, 10, 23);
|
|
mu_assert_notnull (b, "r_buf_new_slice failed");
|
|
|
|
ut64 buf_sz = r_buf_size (b);
|
|
mu_assert_eq (buf_sz, 23, "file size should be computed");
|
|
|
|
int r = r_buf_read_at (b, 0, buffer, 23);
|
|
mu_assert_eq (r, 23, "r_buf_read_at failed");
|
|
mu_assert_memeq (buffer, (ut8 *)"Something To\nSay Here..", 23, "r_buf_read_at has corrupted content");
|
|
|
|
r_buf_seek (b, 3, R_BUF_SET);
|
|
r = r_buf_read (b, buffer, 3);
|
|
mu_assert_eq (r, 3, "only 3 read");
|
|
mu_assert_memeq (buffer, (ut8 *)"eth", 3, "base should be considered");
|
|
|
|
r = r_buf_read (b, buffer, 40);
|
|
mu_assert_eq (r, 23 - 6, "consider limit");
|
|
|
|
bool res = r_buf_resize (b, 30);
|
|
mu_assert ("file should be resized", res);
|
|
buf_sz = r_buf_size (b);
|
|
mu_assert_eq (buf_sz, 30, "file size should be 30");
|
|
|
|
// Cleanup
|
|
r_buf_free (b);
|
|
r_buf_free (buf);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_buf_get_string(void) {
|
|
ut8 *ch = malloc (128);
|
|
memset (ch, 'A', 127);
|
|
ch[127] = '\0';
|
|
RBuffer *b = r_buf_new_with_bytes (ch, 128);
|
|
char *s = r_buf_get_string (b, 100);
|
|
mu_assert_streq (s, (char *)ch + 100, "the string is the same");
|
|
free (s);
|
|
s = r_buf_get_string (b, 0);
|
|
mu_assert_streq (s, (char *)ch, "the string is the same");
|
|
free (s);
|
|
s = r_buf_get_string (b, 127);
|
|
mu_assert_streq (s, "\x00", "the string is empty");
|
|
free (s);
|
|
r_buf_free (b);
|
|
free (ch);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_buf_get_string_nothing(void) {
|
|
RBuffer *b = r_buf_new_with_bytes ((ut8 *)"\x33\x22", 2);
|
|
char *s = r_buf_get_string (b, 0);
|
|
mu_assert_null (s, "there is no string in the buffer (no null terminator)");
|
|
r_buf_append_bytes (b, (ut8 *)"\x00", 1);
|
|
s = r_buf_get_string (b, 0);
|
|
mu_assert_streq (s, "\x33\x22", "now there is a string because of the null terminator");
|
|
free (s);
|
|
r_buf_free (b);
|
|
mu_end;
|
|
}
|
|
|
|
bool test_r_buf_slice_too_big(void) {
|
|
RBuffer *buf = r_buf_new_with_bytes ((ut8 *)"AAAA", 4);
|
|
RBuffer *sl = r_buf_new_slice (buf, 1, 5);
|
|
ut64 sz = r_buf_size (sl);
|
|
mu_assert_eq (sz, 3, "the size cannot be more than the original buffer");
|
|
r_buf_resize (sl, 1);
|
|
sz = r_buf_size (sl);
|
|
mu_assert_eq (sz, 1, "it should be shrinked to 1 byte");
|
|
bool res = r_buf_resize (sl, 7);
|
|
mu_assert ("the resize should be successful", res);
|
|
sz = r_buf_size (sl);
|
|
mu_assert_eq (sz, 3, "but it should just use the biggest value");
|
|
r_buf_free (sl);
|
|
r_buf_free (buf);
|
|
mu_end;
|
|
}
|
|
|
|
int all_tests() {
|
|
mu_run_test (test_r_buf_file);
|
|
mu_run_test (test_r_buf_bytes);
|
|
mu_run_test (test_r_buf_mmap);
|
|
mu_run_test (test_r_buf_with_buf);
|
|
mu_run_test (test_r_buf_slice);
|
|
mu_run_test (test_r_buf_io);
|
|
mu_run_test (test_r_buf_sparse);
|
|
mu_run_test (test_r_buf_sparse2);
|
|
mu_run_test (test_r_buf_bytes_steal);
|
|
mu_run_test (test_r_buf_format);
|
|
mu_run_test (test_r_buf_get_string);
|
|
mu_run_test (test_r_buf_get_string_nothing);
|
|
mu_run_test (test_r_buf_slice_too_big);
|
|
return tests_passed != tests_run;
|
|
}
|
|
|
|
int main(int argc, char **argv) {
|
|
return all_tests();
|
|
}
|