#include "amduat/enc/asl_tgk_exec_plan.h" #include #include #include #include #include enum { k_plan_header_size = 8, k_params_size = 65, k_operator_size = 129, k_input_offset = 97 }; static void store_u32_le(uint8_t *out, uint32_t value) { out[0] = (uint8_t)(value & 0xffu); out[1] = (uint8_t)((value >> 8) & 0xffu); out[2] = (uint8_t)((value >> 16) & 0xffu); out[3] = (uint8_t)((value >> 24) & 0xffu); } static void fill_default_params(amduat_asl_tgk_exec_operator_params_t *params) { memset(params, 0, sizeof(*params)); params->tgk_traversal.direction = 1u; } static int test_round_trip(void) { amduat_asl_tgk_exec_operator_def_t operators[2]; amduat_asl_tgk_exec_plan_t plan; amduat_octets_t encoded; amduat_asl_tgk_exec_plan_t decoded; int exit_code = 1; memset(operators, 0, sizeof(operators)); fill_default_params(&operators[0].params); fill_default_params(&operators[1].params); operators[0].op_id = 1u; operators[0].op_type = AMDUAT_ASL_TGK_EXEC_OP_SEGMENT_SCAN; operators[0].flags = AMDUAT_ASL_TGK_EXEC_OP_FLAG_NONE; operators[0].snapshot.logseq_min = 0u; operators[0].snapshot.logseq_max = 10u; operators[0].input_count = 0u; operators[1].op_id = 2u; operators[1].op_type = AMDUAT_ASL_TGK_EXEC_OP_TGK_TRAVERSAL; operators[1].flags = AMDUAT_ASL_TGK_EXEC_OP_FLAG_PARALLEL; operators[1].snapshot.logseq_min = 1u; operators[1].snapshot.logseq_max = 10u; operators[1].params.tgk_traversal.direction = 2u; operators[1].params.tgk_traversal.traversal_depth = 3u; operators[1].params.tgk_traversal.start_node_id = 42u; operators[1].input_count = 1u; operators[1].inputs[0] = 1u; plan.plan_version = AMDUAT_ASL_TGK_EXEC_PLAN_VERSION; plan.operator_count = 2u; plan.operators = operators; if (!amduat_enc_asl_tgk_exec_plan_encode_v1(&plan, &encoded)) { fprintf(stderr, "encode failed\n"); return exit_code; } if (!amduat_enc_asl_tgk_exec_plan_decode_v1(encoded, &decoded)) { fprintf(stderr, "decode failed\n"); goto cleanup; } if (decoded.plan_version != plan.plan_version || decoded.operator_count != plan.operator_count) { fprintf(stderr, "header mismatch\n"); goto cleanup_decoded; } if (decoded.operators[1].op_type != operators[1].op_type || decoded.operators[1].params.tgk_traversal.direction != 2u || decoded.operators[1].params.tgk_traversal.traversal_depth != 3u || decoded.operators[1].params.tgk_traversal.start_node_id != 42u) { fprintf(stderr, "operator fields mismatch\n"); goto cleanup_decoded; } if (decoded.operators[1].input_count != 1u || decoded.operators[1].inputs[0] != 1u) { fprintf(stderr, "inputs mismatch\n"); goto cleanup_decoded; } exit_code = 0; cleanup_decoded: amduat_enc_asl_tgk_exec_plan_free(&decoded); cleanup: free((void *)encoded.data); return exit_code; } static int test_invalid_input_ref(void) { amduat_asl_tgk_exec_operator_def_t operators[1]; amduat_asl_tgk_exec_plan_t plan; amduat_octets_t encoded; int exit_code = 1; size_t op_offset; size_t input_offset; memset(operators, 0, sizeof(operators)); fill_default_params(&operators[0].params); operators[0].op_id = 1u; operators[0].op_type = AMDUAT_ASL_TGK_EXEC_OP_SEGMENT_SCAN; operators[0].input_count = 0u; plan.plan_version = AMDUAT_ASL_TGK_EXEC_PLAN_VERSION; plan.operator_count = 1u; plan.operators = operators; if (!amduat_enc_asl_tgk_exec_plan_encode_v1(&plan, &encoded)) { fprintf(stderr, "encode failed\n"); return exit_code; } op_offset = k_plan_header_size; if (encoded.len < op_offset + k_input_offset + 4u) { fprintf(stderr, "encoded buffer too small\n"); goto cleanup; } store_u32_le((uint8_t *)encoded.data + op_offset + k_params_size + 28u, 1u); input_offset = op_offset + k_input_offset; store_u32_le((uint8_t *)encoded.data + input_offset, 99u); if (amduat_enc_asl_tgk_exec_plan_decode_v1(encoded, &(amduat_asl_tgk_exec_plan_t){0})) { fprintf(stderr, "decode unexpectedly succeeded\n"); goto cleanup; } exit_code = 0; cleanup: free((void *)encoded.data); return exit_code; } static int test_truncated_decode(void) { amduat_asl_tgk_exec_operator_def_t operators[1]; amduat_asl_tgk_exec_plan_t plan; amduat_octets_t encoded; amduat_octets_t truncated; int exit_code = 1; memset(operators, 0, sizeof(operators)); fill_default_params(&operators[0].params); operators[0].op_id = 1u; operators[0].op_type = AMDUAT_ASL_TGK_EXEC_OP_SEGMENT_SCAN; plan.plan_version = AMDUAT_ASL_TGK_EXEC_PLAN_VERSION; plan.operator_count = 1u; plan.operators = operators; if (!amduat_enc_asl_tgk_exec_plan_encode_v1(&plan, &encoded)) { fprintf(stderr, "encode failed\n"); return exit_code; } truncated = amduat_octets(encoded.data, encoded.len - 1u); if (amduat_enc_asl_tgk_exec_plan_decode_v1(truncated, &(amduat_asl_tgk_exec_plan_t){0})) { fprintf(stderr, "decode unexpectedly succeeded\n"); goto cleanup; } exit_code = 0; cleanup: free((void *)encoded.data); return exit_code; } static int test_invalid_input_count(void) { amduat_asl_tgk_exec_operator_def_t operators[1]; amduat_asl_tgk_exec_plan_t plan; amduat_octets_t encoded; int exit_code = 1; size_t op_offset; size_t input_count_offset; memset(operators, 0, sizeof(operators)); fill_default_params(&operators[0].params); operators[0].op_id = 1u; operators[0].op_type = AMDUAT_ASL_TGK_EXEC_OP_SEGMENT_SCAN; plan.plan_version = AMDUAT_ASL_TGK_EXEC_PLAN_VERSION; plan.operator_count = 1u; plan.operators = operators; if (!amduat_enc_asl_tgk_exec_plan_encode_v1(&plan, &encoded)) { fprintf(stderr, "encode failed\n"); return exit_code; } op_offset = k_plan_header_size; input_count_offset = op_offset + k_params_size + 28u; if (encoded.len < input_count_offset + 4u) { fprintf(stderr, "encoded buffer too small\n"); goto cleanup; } store_u32_le((uint8_t *)encoded.data + input_count_offset, 9u); if (amduat_enc_asl_tgk_exec_plan_decode_v1(encoded, &(amduat_asl_tgk_exec_plan_t){0})) { fprintf(stderr, "decode unexpectedly succeeded\n"); goto cleanup; } exit_code = 0; cleanup: free((void *)encoded.data); return exit_code; } int main(void) { if (test_round_trip() != 0) { return 1; } if (test_invalid_input_ref() != 0) { return 1; } if (test_truncated_decode() != 0) { return 1; } return test_invalid_input_count(); }