// Copyright (c) 2023 Yuki Kimoto
// MIT License
// Windows 8.1+
#define _WIN32_WINNT 0x0603
#include "spvm_native.h"
#include <assert.h>
#if defined(_WIN32)
#include <winsock2.h>
#else
#include <poll.h>
#endif
static const char* FILE_NAME = "Sys/Poll/PollfdArray";
int32_t SPVM__Sys__Poll__PollfdArray___maybe_extend(SPVM_ENV* env, SPVM_VALUE* stack);
int32_t SPVM__Sys__Poll__PollfdArray__new(SPVM_ENV* env, SPVM_VALUE* stack) {
int32_t error_id = 0;
int32_t length = stack[0].ival;
if (!(length >= 0)) {
return env->die(env, stack, "The length $length must be greater than or equal to 0.", __func__, FILE_NAME, __LINE__);
}
int32_t capacity = stack[1].ival;
if (capacity < 0) {
capacity = length;
if (capacity == 0) {
capacity = 1;
}
}
struct pollfd* pollfds = env->new_memory_block(env, stack, sizeof(struct pollfd) * capacity);
void* obj_self = env->new_pointer_object_by_name(env, stack, "Sys::Poll::PollfdArray", pollfds, &error_id, __func__, FILE_NAME, __LINE__);
if (error_id) { return error_id; }
env->set_field_int_by_name(env, stack, obj_self, "length", length, &error_id, __func__, FILE_NAME, __LINE__);
if (error_id) { return error_id; }
env->set_field_int_by_name(env, stack, obj_self, "capacity", capacity, &error_id, __func__, FILE_NAME, __LINE__);
if (error_id) { return error_id; }
stack[0].oval = obj_self;
return 0;
}
int32_t SPVM__Sys__Poll__PollfdArray__DESTROY(SPVM_ENV* env, SPVM_VALUE* stack) {
void* obj_self = stack[0].oval;
struct pollfd* pollfds = env->get_pointer(env, stack, obj_self);
assert(pollfds);
env->free_memory_block(env, stack, pollfds);
env->set_pointer(env, stack, obj_self, NULL);
return 0;
}
int32_t SPVM__Sys__Poll__PollfdArray__fd(SPVM_ENV* env, SPVM_VALUE* stack) {
int32_t error_id = 0;
void* obj_self = stack[0].oval;
int32_t index = stack[1].ival;
int32_t length = env->get_field_int_by_name(env, stack, obj_self, "length", &error_id, __func__, FILE_NAME, __LINE__);
if (error_id) { return error_id; }
if (!(index >= 0)) {
return env->die(env, stack, "The index $index must be greater than or equal to 0.", __func__, FILE_NAME, __LINE__);
}
if (!(index < length)) {
return env->die(env, stack, "The index $index must be less than the value of the length field.", __func__, FILE_NAME, __LINE__);
}
struct pollfd* pollfds = env->get_pointer(env, stack, obj_self);
int32_t fd = pollfds[index].fd;
stack[0].ival = fd;
return 0;
}
int32_t SPVM__Sys__Poll__PollfdArray__set_fd(SPVM_ENV* env, SPVM_VALUE* stack) {
int32_t error_id = 0;
void* obj_self = stack[0].oval;
int32_t index = stack[1].ival;
int32_t length = env->get_field_int_by_name(env, stack, obj_self, "length", &error_id, __func__, FILE_NAME, __LINE__);
if (error_id) { return error_id; }
if (!(index >= 0)) {
return env->die(env, stack, "The index $index must be greater than or equal to 0.", __func__, FILE_NAME, __LINE__);
}
if (!(index < length)) {
return env->die(env, stack, "The index $index must be less than the value of the length field.", __func__, FILE_NAME, __LINE__);
}
int32_t fd = stack[2].ival;
struct pollfd* pollfds = env->get_pointer(env, stack, obj_self);
pollfds[index].fd = fd;
return 0;
}
int32_t SPVM__Sys__Poll__PollfdArray__events(SPVM_ENV* env, SPVM_VALUE* stack) {
int32_t error_id = 0;
void* obj_self = stack[0].oval;
int32_t index = stack[1].ival;
if (!(index >= 0)) {
return env->die(env, stack, "The index $index must be greater than or equal to 0.", __func__, FILE_NAME, __LINE__);
}
int32_t length = env->get_field_int_by_name(env, stack, obj_self, "length", &error_id, __func__, FILE_NAME, __LINE__);
if (error_id) { return error_id; }
if (!(index < length)) {
return env->die(env, stack, "The index $index must be less than the value of the length field.", __func__, FILE_NAME, __LINE__);
}
struct pollfd* pollfds = env->get_pointer(env, stack, obj_self);
int16_t events = pollfds[index].events;
stack[0].ival = events;
return 0;
}
int32_t SPVM__Sys__Poll__PollfdArray__set_events(SPVM_ENV* env, SPVM_VALUE* stack) {
int32_t error_id = 0;
void* obj_self = stack[0].oval;
int32_t index = stack[1].ival;
if (!(index >= 0)) {
return env->die(env, stack, "The index $index must be greater than or equal to 0.", __func__, FILE_NAME, __LINE__);
}
int32_t length = env->get_field_int_by_name(env, stack, obj_self, "length", &error_id, __func__, FILE_NAME, __LINE__);
if (error_id) { return error_id; }
if (!(index < length)) {
return env->die(env, stack, "The index $index must be less than the value of the length field.", __func__, FILE_NAME, __LINE__);
}
int16_t events = stack[2].ival;
struct pollfd* pollfds = env->get_pointer(env, stack, obj_self);
pollfds[index].events = events;
return 0;
}
int32_t SPVM__Sys__Poll__PollfdArray__revents(SPVM_ENV* env, SPVM_VALUE* stack) {
int32_t error_id = 0;
void* obj_self = stack[0].oval;
int32_t index = stack[1].ival;
if (!(index >= 0)) {
return env->die(env, stack, "The index $index must be greater than or equal to 0.", __func__, FILE_NAME, __LINE__);
}
int32_t length = env->get_field_int_by_name(env, stack, obj_self, "length", &error_id, __func__, FILE_NAME, __LINE__);
if (error_id) { return error_id; }
if (!(index < length)) {
return env->die(env, stack, "The index $index must be less than the value of the length field.", __func__, FILE_NAME, __LINE__);
}
struct pollfd* pollfds = env->get_pointer(env, stack, obj_self);
int16_t revents = pollfds[index].revents;
stack[0].ival = revents;
return 0;
}
int32_t SPVM__Sys__Poll__PollfdArray__set_revents(SPVM_ENV* env, SPVM_VALUE* stack) {
int32_t error_id = 0;
void* obj_self = stack[0].oval;
int32_t index = stack[1].ival;
if (!(index >= 0)) {
return env->die(env, stack, "The index $index must be greater than or equal to 0.", __func__, FILE_NAME, __LINE__);
}
int32_t length = env->get_field_int_by_name(env, stack, obj_self, "length", &error_id, __func__, FILE_NAME, __LINE__);
if (error_id) { return error_id; }
if (!(index < length)) {
return env->die(env, stack, "The index $index must be less than the value of the length field.", __func__, FILE_NAME, __LINE__);
}
int16_t revents = stack[2].ival;
struct pollfd* pollfds = env->get_pointer(env, stack, obj_self);
pollfds[index].revents = revents;
return 0;
}
int32_t SPVM__Sys__Poll__PollfdArray__push(SPVM_ENV* env, SPVM_VALUE* stack) {
int32_t error_id = 0;
void* obj_self = stack[0].oval;
int32_t fd = stack[1].ival;
int32_t length_field = env->get_field_int_by_name(env, stack, obj_self, "length", &error_id, __func__, FILE_NAME, __LINE__);
if (error_id) { return error_id; }
int32_t new_length = length_field + 1;
stack[0].oval = obj_self;
stack[1].ival = new_length;
SPVM__Sys__Poll__PollfdArray___maybe_extend(env, stack);
struct pollfd* pollfds = env->get_pointer(env, stack, obj_self);
pollfds[new_length - 1].fd = fd;
env->set_field_int_by_name(env, stack, obj_self, "length", new_length, &error_id, __func__, FILE_NAME, __LINE__);
if (error_id) { return error_id; }
return 0;
}
int32_t SPVM__Sys__Poll__PollfdArray__remove(SPVM_ENV* env, SPVM_VALUE* stack) {
int32_t error_id = 0;
void* obj_self = stack[0].oval;
int32_t index = stack[1].ival;
int32_t length_field = env->get_field_int_by_name(env, stack, obj_self, "length", &error_id, __func__, FILE_NAME, __LINE__);
if (error_id) { return error_id; }
if (!(index >= 0)) {
return env->die(env, stack, "The index $index must be greater than or equal to 0.", __func__, FILE_NAME, __LINE__);
}
if (!(index < length_field)) {
return env->die(env, stack, "The index $index must be less than the value of length field.", __func__, FILE_NAME, __LINE__);
}
struct pollfd* pollfds = env->get_pointer(env, stack, obj_self);
int32_t move_length = length_field - index - 1;
memcpy((void*)(pollfds + index), (void*)(pollfds + index + 1), sizeof (struct pollfd) * move_length);
memset(&pollfds[length_field - 1], 0, sizeof(struct pollfd));
length_field--;
env->set_field_int_by_name(env, stack, obj_self, "length", length_field, &error_id, __func__, FILE_NAME, __LINE__);
if (error_id) { return error_id; }
return 0;
}
int32_t SPVM__Sys__Poll__PollfdArray___maybe_extend(SPVM_ENV* env, SPVM_VALUE* stack) {
int32_t error_id = 0;
void* obj_self = stack[0].oval;
int32_t min_capacity = stack[1].ival;
int32_t self_capacity = env->get_field_int_by_name(env, stack, obj_self, "capacity", &error_id, __func__, FILE_NAME, __LINE__);
if (error_id) { return error_id; }
if (!(min_capacity > self_capacity)) {
return 0;
}
int32_t new_capacity = min_capacity * 2;
struct pollfd* old_pollfds = env->get_pointer(env, stack, obj_self);
struct pollfd* new_pollfds = env->new_memory_block(env, stack, sizeof(struct pollfd) * new_capacity);
int32_t length_field = env->get_field_int_by_name(env, stack, obj_self, "length", &error_id, __func__, FILE_NAME, __LINE__);
if (error_id) { return error_id; }
memcpy((void*)new_pollfds, (void*)old_pollfds, sizeof(struct pollfd) * length_field);
env->free_memory_block(env, stack, old_pollfds);
env->set_pointer(env, stack, obj_self, new_pollfds);
env->set_field_int_by_name(env, stack, obj_self, "capacity", new_capacity, &error_id, __func__, FILE_NAME, __LINE__);
if (error_id) { return error_id; }
return 0;
}