#include <stdint.h> #include <inttypes.h> #include <stdlib.h> #include <string.h> #include <stdio.h> #include "../../../none/tests/s390x/opcodes.h" /* Define various input buffers. */ /* U+0000 to U+007f: Result is 1 byte for each uint16_t */ uint16_t pattern1[] = { 0x0000, 0x007f, /* corner cases */ 0x0047, 0x0056, 0x0045, 0x0021, 0x007b, 0x003a /* misc */ }; /* U+0080 to U+07ff: Result is 2 bytes for each uint16_t */ uint16_t pattern2[] = { 0x0080, 0x07ff, /* corner cases */ 0x07df, 0x008f, 0x0100, 0x017f, 0x052f, 0x0600, 0x06ff /* misc */ }; /* U+0800 to U+d7ff: Result is 3 bytes for each uint16_t U+dc00 to U+ffff: Result is 3 bytes for each uint16_t */ uint16_t pattern3[] = { 0x0800, 0xd7ff, /* corner cases */ 0xdc00, 0xffff, /* corner cases */ 0x083f, 0x1a21, 0x1b10, 0x2200, 0x225e, 0x22c9, 0xe001 /* misc */ }; /* U+d800 to U+dbff: Result is 4 bytes for each uint16_t pair */ uint16_t pattern4[] = { 0xd800, 0xdc00, /* left corner case */ 0xdbff, 0xdfff, /* right corner case */ 0xdada, 0xdddd, 0xdeaf, 0xdcdc /* misc */ }; void do_cu21(uint8_t *dst, uint64_t dst_len, uint16_t *src, uint64_t src_len) { /* build up the register pairs */ register uint16_t *source asm("4") = src; register uint64_t source_len asm("5") = src_len; register uint8_t *dest asm("2") = dst; register uint64_t dest_len asm("3") = dst_len; asm volatile( CU21(0,2,4) : "+d"(dest), "+d"(source), "+d"(source_len), "+d"(dest_len) : : "memory", "cc"); return; } int main() { /*------------------------------------------------------------*/ /* Write to a too small buffer */ /*------------------------------------------------------------*/ /* Write 2 bytes into buffer of length 1 */ do_cu21(malloc(1), 10, pattern2, 2); // complaint (2 bytes) /* Write 2 bytes into buffer of length 2 */ do_cu21(malloc(2), 10, pattern2, 2); // no complaint /* Write 3 bytes into buffer of length 1 */ do_cu21(malloc(1), 10, pattern3, 2); // 2 complaints (3 = 2+1) /* Write 3 bytes into buffer of length 2 */ do_cu21(malloc(2), 10, pattern3, 2); // complaint (1 byte) /* Write 3 bytes into buffer of length 3 */ do_cu21(malloc(3), 10, pattern3, 2); // no complaint /* Write 4 bytes into buffer of length 1 */ do_cu21(malloc(1), 10, pattern4, 4); // complaint (4 bytes) /* Write 4 bytes into buffer of length 2 */ do_cu21(malloc(2), 10, pattern4, 4); // complaint (4 bytes) /* Write 4 bytes into buffer of length 3 */ do_cu21(malloc(3), 10, pattern4, 4); // complaint (4 bytes) /* Write 4 bytes into buffer of length 4 */ do_cu21(malloc(4), 10, pattern4, 4); // no complaint /*------------------------------------------------------------*/ /* Read uninitialised data */ /*------------------------------------------------------------*/ uint8_t *input = malloc(10); /* Input buffer is completely uninitialised */ do_cu21(malloc(4), 4, (void *)input, 2); // complaint /* Read 2 bytes from input buffer. First byte is uninitialised */ input = malloc(10); input[1] = 0x0; do_cu21(malloc(4), 4, (void *)input, 2); // complaint /* Read 2 bytes from input buffer. Second byte is uninitialised */ input = malloc(10); input[0] = 0x0; do_cu21(malloc(4), 4, (void *)input, 2); // complaint /* Read 2 bytes from input buffer. All bytes are initialised */ input = malloc(10); input[0] = input[1] = 0x0; do_cu21(malloc(4), 4, (void *)input, 2); // no complaint /* Read 4 bytes from input buffer. This iterates once. In the 1st iteration all input bytes are initialised in the 2nd iteration all input bytes are uninitialised. */ input = malloc(10); input[0] = input[1] = 0x0; do_cu21(malloc(4), 4, (void *)input, 4); // complaint /* Write to NULL */ // do_cu21(NULL, 10, pattern1, sizeof pattern1); // complaint return 0; }