| /* -*- Mode: C; tab-width: 8; c-basic-offset: 2; indent-tabs-mode: nil; -*- */ |
| |
| #include "util.h" |
| #include <linux/keyctl.h> |
| |
| typedef uint32_t key_serial_t; |
| |
| /* There was a breaking UAPI change so use this */ |
| struct rr_keyctl_dh_params { |
| __s32 dh_private; |
| __s32 prime; |
| __s32 base; |
| }; |
| |
| int main(void) { |
| char buffer[256]; |
| |
| char* data = "Test Data"; |
| key_serial_t key = syscall(SYS_add_key, "user", "RR Test key", data, |
| strlen(data) + 1, KEY_SPEC_PROCESS_KEYRING); |
| test_assert(0 == syscall(SYS_keyctl, KEYCTL_SETPERM, key, 0x3f3f0000)); |
| long result = |
| syscall(SYS_keyctl, KEYCTL_DESCRIBE, key, buffer, sizeof(buffer)); |
| test_assert(-1 != result); |
| check_data(buffer, result); |
| result = syscall(SYS_keyctl, KEYCTL_READ, key, buffer, sizeof(buffer)); |
| test_assert(-1 != result); |
| check_data(buffer, result); |
| test_assert(0 == memcmp(buffer, data, strlen(data) + 1)); |
| result = |
| syscall(SYS_keyctl, KEYCTL_GET_SECURITY, key, buffer, sizeof(buffer)); |
| test_assert(-1 != result); |
| check_data(buffer, result); |
| |
| test_assert(0 == syscall(SYS_keyctl, KEYCTL_INVALIDATE, key)); |
| |
| #ifdef KEYCTL_DH_COMPUTE |
| uint8_t base[256] = { 1 }; |
| // 'prime' must be at least 1536 bits or we get EINVAL. |
| uint8_t prime[256] = { 7 }; // Not prime but we should be OK |
| uint8_t private = 1; // The world's worst private key |
| base[191] = 'x'; |
| |
| key_serial_t base_key = syscall(SYS_add_key, "user", "base", base, |
| sizeof(base), KEY_SPEC_PROCESS_KEYRING); |
| key_serial_t prime_key = syscall(SYS_add_key, "user", "prime", prime, |
| sizeof(prime), KEY_SPEC_PROCESS_KEYRING); |
| key_serial_t private_key = syscall(SYS_add_key, "user", "private", &private, |
| sizeof(private), KEY_SPEC_PROCESS_KEYRING); |
| |
| struct rr_keyctl_dh_params params = {.dh_private = private_key, |
| .prime = prime_key, |
| .base = base_key }; |
| |
| result = syscall(SYS_keyctl, KEYCTL_DH_COMPUTE, ¶ms, buffer, |
| sizeof(buffer), (void*)0); |
| if (-1 == result) { |
| // This one's rather new. May not be supported. |
| test_assert(errno == EOPNOTSUPP); |
| } else { |
| check_data(buffer, result); |
| test_assert(0 == memcmp(buffer, base, sizeof(base))); |
| } |
| #endif |
| |
| atomic_puts("EXIT-SUCCESS"); |
| return 0; |
| } |