Files
vasilito b9874d0941 feat: USB storage read/write proof + full Red Bear OS tree sync
Add redbear-usb-storage-check in-guest binary that validates USB mass
storage read and write I/O: discovers /scheme/disk/ devices, writes a
test pattern to sector 2048, reads it back, verifies match, restores
original content. Updates test-usb-storage-qemu.sh with write-proof
verification step.

Includes all accumulated Red Bear OS work: kernel patches, relibc
patches, driver infrastructure, DRM/GPU, KDE recipes, firmware,
validation tooling, build system hardening, and documentation.
2026-05-03 23:03:24 +01:00

342 lines
13 KiB
C

#include <malloc.h> /* for pvalloc() */
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <stddef.h> /* for size_t */
#include <stdint.h> /* for SIZE_MAX */
#include <string.h> /* for strerror() */
#include <unistd.h> /* for sysconf() */
#include "test_helpers.h"
/* For regular allocations that should succeed without particular
* alignment requirements. */
void test_non_null(void *ptr, int error_val) {
if (ptr != NULL) {
// Constant output for successful case
printf("pointer: (not NULL), ");
}
else {
printf("pointer: %p, ", ptr);
}
printf("error value: %d = %s\n",
error_val, strerror(error_val));
}
/* For testing functions that should return pointers with a particular
* alignment (successful case). */
void test_valid_aligned(void *ptr, size_t alignment, int error_val) {
/* Cast to uintptr_t to allow taking modulo of address. The
* uintptr_t type is guaranteed to be able to hold any valid object
* address. */
uintptr_t ptr_alignment_rem = (uintptr_t)ptr % (uintptr_t)alignment;
if (ptr != NULL && ptr_alignment_rem == 0) {
// Constant output for successful case
printf("pointer: (alignment OK), ");
}
else {
printf("pointer: %p, ", ptr);
}
printf("error value: %d = %s\n",
error_val, strerror(error_val));
}
/* For testing functions that should return pointers with a particular
* alignment. With invalid alignment, we expect constant output (a NULL
* pointer and EINVAL). */
void test_invalid_aligned(void *ptr, int error_val) {
printf("pointer: %p, error value: %d = %s\n",
ptr, error_val, strerror(error_val));
}
/* For testing size-0 allocation requests. */
void test_size_zero(void *ptr, size_t alignment, int error_val) {
/* Facilitates checking alignment upon non-NULL return */
uintptr_t ptr_alignment_rem = (uintptr_t)ptr % (uintptr_t)alignment;
/* For allocation functions, POSIX permits returning either a NULL
* pointer and optionally an implementation-defined error value, or
* succeeding with a non-NULL pointer. */
if (ptr == NULL || (ptr_alignment_rem == 0 && error_val == 0)) {
// Constant output for successful case
printf("(OK)\n");
}
else {
printf("pointer: %p, error value: %d = %s\n",
ptr, error_val, strerror(error_val));
}
}
/* For cases where we expect allocation to fail, returning a NULL
* pointer and indicating ENOMEM. */
void test_cannot_alloc(void *ptr, int error_val) {
printf("pointer: %p, error value: %d = %s\n",
ptr, error_val, strerror(error_val));
}
int main(void) {
size_t sample_alloc_size = 256;
size_t sample_realloc_size = sample_alloc_size + 1;
/* ensure values are mapped to variables */
size_t zero_size = 0;
size_t max_size = SIZE_MAX;
size_t page_size = (size_t)sysconf(_SC_PAGESIZE);
size_t aligned_alloc_alignment = 128;
size_t aligned_alloc_goodsize = 256;
size_t aligned_alloc_badsize = 257;
size_t nonpow2_mul_voidptr_size = 3*sizeof(void *);
size_t pow2_mul_voidptr_size = 4*sizeof(void *);
size_t i;
errno = 0;
char * ptr_zerosize_malloc = (char *)malloc(zero_size);
int malloc_zerosize_errno = errno;
printf("malloc (size 0): ");
test_size_zero(ptr_zerosize_malloc, 1, malloc_zerosize_errno);
free(ptr_zerosize_malloc);
errno = 0;
char * ptr_malloc = (char *)malloc(sample_alloc_size);
int malloc_errno = errno;
printf("malloc: ");
test_non_null(ptr_malloc, malloc_errno);
for(i = 0; i < sample_alloc_size; i++) {
ptr_malloc[i] = (char)i;
}
free(ptr_malloc);
errno = 0;
char * ptr_malloc_maxsize = (char *)malloc(max_size);
int malloc_maxsize_errno = errno;
printf("malloc (SIZE_MAX): ");
test_cannot_alloc(ptr_malloc_maxsize, malloc_maxsize_errno);
free(ptr_malloc_maxsize);
errno = 0;
char * ptr_zerosize_calloc = (char *)calloc(zero_size, 1);
int calloc_zerosize_errno = errno;
printf("calloc (size 0): ");
test_size_zero(ptr_zerosize_calloc, 1, calloc_zerosize_errno);
free(ptr_zerosize_calloc);
errno = 0;
char * ptr_calloc = (char *)calloc(sample_alloc_size, 1);
int calloc_errno = errno;
printf("calloc: ");
test_non_null(ptr_calloc, calloc_errno);
for(i = 0; i < sample_alloc_size; i++) {
ptr_calloc[i] = (char)i;
}
free(ptr_calloc);
errno = 0;
char * ptr_calloc_overflow = (char *)calloc(max_size, max_size);
int calloc_overflow_errno = errno;
printf("calloc (overflowing): ");
test_cannot_alloc(ptr_calloc_overflow, calloc_overflow_errno);
free(ptr_calloc_overflow);
char * ptr_realloc_size0 = (char *)malloc(sample_alloc_size);
errno = 0;
ptr_realloc_size0 = (char *)realloc(ptr_realloc_size0, zero_size);
int realloc_size0_errno = errno;
printf("realloc (size 0): ");
test_size_zero(ptr_realloc_size0, 1, realloc_size0_errno);
free(ptr_realloc_size0);
char * ptr_realloc = (char *)malloc(sample_alloc_size);
errno = 0;
ptr_realloc = (char *)realloc(ptr_realloc, sample_realloc_size);
int realloc_errno = errno;
printf("realloc: ");
test_non_null(ptr_realloc, realloc_errno);
for(i = 0; i < sample_realloc_size; i++) {
ptr_realloc[i] = (char)i;
}
free(ptr_realloc);
char * ptr_realloc_maxsize = (char *)malloc(sample_alloc_size);
errno = 0;
ptr_realloc_maxsize = (char *)realloc(ptr_realloc_maxsize, max_size);
int realloc_maxsize_errno = errno;
printf("realloc (SIZE_MAX): ");
test_cannot_alloc(ptr_realloc_maxsize, realloc_maxsize_errno);
free(ptr_realloc_maxsize);
errno = 0;
char * ptr_reallocarray_maxsize = (char *)malloc(sample_alloc_size);
ptr_reallocarray_maxsize = (char *)reallocarray(ptr_reallocarray_maxsize, 2, sample_alloc_size);
int reallocarray_errno = errno;
printf("reallocarray: ");
test_non_null(ptr_reallocarray_maxsize, reallocarray_errno);
for(i = 0; i < sample_realloc_size; i++) {
ptr_realloc[i] = (char)i;
}
errno = 0;
ptr_reallocarray_maxsize = (char *)reallocarray(ptr_reallocarray_maxsize, 2, max_size);
reallocarray_errno = errno;
printf("reallocarray (SIZE_MAX): ");
test_cannot_alloc(ptr_reallocarray_maxsize, reallocarray_errno);
free(ptr_reallocarray_maxsize);
// Warning for deprecated functions is expected so silence -Werror
#pragma GCC diagnostic push
#pragma GCC diagnostic ignored "-Wdeprecated-declarations"
errno = 0;
char * ptr_memalign_size0 = (char *)memalign(aligned_alloc_alignment, zero_size);
int memalign_size0_errno = errno;
printf("memalign (size 0): ");
test_size_zero(ptr_memalign_size0, aligned_alloc_alignment, memalign_size0_errno);
free(ptr_memalign_size0);
errno = 0;
char * ptr_memalign = (char *)memalign(aligned_alloc_alignment, sample_alloc_size);
int memalign_errno = errno;
printf("memalign: ");
test_valid_aligned(ptr_memalign, aligned_alloc_alignment, memalign_errno);
for(i = 0; i < sample_alloc_size; i++) {
ptr_memalign[i] = (char)i;
}
free(ptr_memalign);
errno = 0;
char * ptr_memalign_maxsize = (char *)memalign(aligned_alloc_alignment, max_size);
int memalign_maxsize_errno = errno;
printf("memalign (SIZE_MAX): ");
test_cannot_alloc(ptr_memalign_maxsize, memalign_maxsize_errno);
free(ptr_memalign_maxsize);
errno = 0;
char * ptr_memalign_align0 = (char *)memalign(0, sample_alloc_size);
int memalign_align0_errno = errno;
printf("memalign (alignment 0): ");
test_invalid_aligned(ptr_memalign_align0, memalign_align0_errno);
free(ptr_memalign_align0);
errno = 0;
char * ptr_memalign_align3 = (char *)memalign(3, sample_alloc_size);
int memalign_align3_errno = errno;
printf("memalign (alignment 3): ");
test_invalid_aligned(ptr_memalign_align3, memalign_align3_errno);
free(ptr_memalign_align3);
#pragma GCC diagnostic pop
errno = 0;
char * ptr_aligned_alloc_goodsize = (char *)aligned_alloc(aligned_alloc_alignment, aligned_alloc_goodsize);
int aligned_alloc_goodsize_errno = errno;
printf("aligned_alloc (size %% alignment == 0): ");
test_valid_aligned(ptr_aligned_alloc_goodsize, aligned_alloc_alignment, aligned_alloc_goodsize_errno);
free(ptr_aligned_alloc_goodsize);
errno = 0;
char * ptr_aligned_alloc_badsize = (char *)aligned_alloc(aligned_alloc_alignment, aligned_alloc_badsize);
int aligned_alloc_badsize_errno = errno;
printf("aligned_alloc (size %% alignment != 0): ");
test_invalid_aligned(ptr_aligned_alloc_badsize, aligned_alloc_badsize_errno);
free(ptr_aligned_alloc_badsize);
// Warning for deprecated functions is expected so silence -Werror
#pragma GCC diagnostic push
#pragma GCC diagnostic ignored "-Wdeprecated-declarations"
errno = 0;
char * ptr_valloc_size0 = (char *)valloc(zero_size);
int valloc_size0_errno = errno;
printf("valloc (size 0): ");
test_size_zero(ptr_valloc_size0, page_size, valloc_size0_errno);
free(ptr_valloc_size0);
errno = 0;
char * ptr_valloc = (char *)valloc(sample_alloc_size);
int valloc_errno = errno;
printf("valloc: ");
test_valid_aligned(ptr_valloc, page_size, valloc_errno);
free(ptr_valloc);
errno = 0;
char * ptr_valloc_maxsize = (char *)valloc(max_size);
int valloc_maxsize_errno = errno;
printf("valloc (SIZE_MAX): ");
test_cannot_alloc(ptr_valloc_maxsize, valloc_maxsize_errno);
free(ptr_valloc_maxsize);
errno = 0;
void * ptr_posix_memalign = NULL;
int posix_memalign_return = posix_memalign(&ptr_posix_memalign, pow2_mul_voidptr_size, sample_alloc_size);
printf("posix_memalign: ");
test_valid_aligned(ptr_posix_memalign, pow2_mul_voidptr_size, posix_memalign_return);
free(ptr_posix_memalign);
errno = 0;
void * ptr_posix_memalign_align0 = NULL;
int posix_memalign_align0_return = posix_memalign(&ptr_posix_memalign_align0, zero_size, sample_alloc_size);
printf("posix_memalign (alignment 0): ");
test_invalid_aligned(ptr_posix_memalign_align0, posix_memalign_align0_return);
free(ptr_posix_memalign_align0);
errno = 0;
void * ptr_posix_memalign_nonpow2mul = NULL;
int posix_memalign_nonpow2mul_return = posix_memalign(&ptr_posix_memalign_nonpow2mul, nonpow2_mul_voidptr_size, sample_alloc_size);
printf("posix_memalign (non-power-of-two multiple of sizeof(void *)): ");
test_invalid_aligned(ptr_posix_memalign_nonpow2mul, posix_memalign_nonpow2mul_return);
free(ptr_posix_memalign_nonpow2mul);
errno = 0;
void * ptr_posix_memalign_size0 = NULL;
int posix_memalign_size0_return = posix_memalign(&ptr_posix_memalign_size0, pow2_mul_voidptr_size, zero_size);
printf("posix_memalign (size 0): ");
test_size_zero(ptr_posix_memalign_size0, pow2_mul_voidptr_size, posix_memalign_size0_return);
free(ptr_posix_memalign_size0);
errno = 0;
void * ptr_posix_memalign_maxsize = NULL;
int posix_memalign_maxsize_return = posix_memalign(&ptr_posix_memalign_maxsize, pow2_mul_voidptr_size, max_size);
printf("posix_memalign (SIZE_MAX): ");
test_cannot_alloc(ptr_posix_memalign_maxsize, posix_memalign_maxsize_return);
free(ptr_posix_memalign_maxsize);
errno = 0;
char * ptr_pvalloc_size0 = (char *)pvalloc(zero_size);
int pvalloc_size0_errno = errno;
printf("pvalloc (size 0): ");
test_size_zero(ptr_pvalloc_size0, page_size, pvalloc_size0_errno);
free(ptr_pvalloc_size0);
errno = 0;
char * ptr_pvalloc = (char *)pvalloc(sample_alloc_size);
int pvalloc_errno = errno;
printf("pvalloc: ");
test_valid_aligned(ptr_pvalloc, page_size, pvalloc_errno);
/* Check that the alloc size is rounded up to nearest page-size
* multiple */
for(i = 0; i < page_size; i++) {
ptr_pvalloc[i] = (char)i;
}
free(ptr_pvalloc);
errno = 0;
char * ptr_pvalloc_multipage = (char *)pvalloc(page_size + 1);
int pvalloc_multipage_errno = errno;
printf("pvalloc (2 pages): ");
test_valid_aligned(ptr_pvalloc_multipage, page_size, pvalloc_multipage_errno);
/* Check that the alloc size is rounded up to nearest page-size
* multiple */
for(i = 0; i < 2*page_size; i++) {
ptr_pvalloc_multipage[i] = (char)i;
}
free(ptr_pvalloc_multipage);
errno = 0;
char * ptr_pvalloc_maxsize = (char *)pvalloc(max_size);
int pvalloc_maxsize_errno = errno;
printf("pvalloc (SIZE_MAX): ");
test_cannot_alloc(ptr_pvalloc_maxsize, pvalloc_maxsize_errno);
free(ptr_pvalloc_maxsize);
#pragma GCC diagnostic pop
}