ardix/kernel/device.c

99 lines
2.4 KiB
C
Raw Normal View History

/* See the end of this file for copyright, license, and warranty information. */
#include <ardix/device.h>
#include <ardix/kent.h>
#include <ardix/list.h>
#include <ardix/malloc.h>
#include <ardix/types.h>
#include <errno.h>
#include <stddef.h>
struct kent *devices_kent = NULL;
static void devices_destroy(struct kent *kent)
2021-02-17 15:01:15 +01:00
{
/* should never be executed because the root devices kent is immortal */
free(kent);
2021-02-17 15:01:15 +01:00
}
/** Initialize the devices subsystem. */
int devices_init(void)
{
if (devices_kent != NULL)
return -EEXIST;
devices_kent = malloc(sizeof(*devices_kent));
if (devices_kent == NULL)
return -ENOMEM;
devices_kent->parent = kent_root;
2021-07-31 15:58:29 +02:00
devices_kent->destroy = devices_destroy;
2021-02-17 15:01:15 +01:00
return kent_init(devices_kent);
}
static void device_destroy(struct kent *kent)
{
struct device *dev = kent_to_device(kent);
free(dev);
}
int device_init(struct device *dev)
{
2021-07-31 15:58:29 +02:00
if (dev->kent.destroy == NULL)
dev->kent.destroy = device_destroy;
if (dev->kent.parent == NULL)
dev->kent.parent = devices_kent;
2021-08-01 04:22:09 +02:00
mutex_init(&dev->lock);
return kent_init(&dev->kent);
}
2021-08-04 03:29:52 +02:00
static void device_kevent_destroy(struct kent *kent)
{
struct kevent *event = container_of(kent, struct kevent, kent);
struct device_kevent *device_kevent = container_of(event, struct device_kevent, kevent);
2021-08-04 03:29:52 +02:00
free(device_kevent);
}
struct device_kevent *device_kevent_create(struct device *device, enum device_channel channel)
{
2021-08-10 00:44:36 +02:00
struct device_kevent *event = atomic_malloc(sizeof(*event));
2021-08-04 03:29:52 +02:00
if (event == NULL)
return NULL;
event->channel = channel;
event->kevent.kind = KEVENT_DEVICE;
2021-08-04 03:29:52 +02:00
event->kevent.kent.parent = &device->kent;
event->kevent.kent.destroy = device_kevent_destroy;
int err = kent_init(&event->kevent.kent);
2021-08-04 03:29:52 +02:00
if (err) {
free(event);
event = NULL;
}
return event;
}
2021-08-04 15:46:51 +02:00
void device_kevent_create_and_dispatch(struct device *device, enum device_channel channel)
{
struct device_kevent *event = device_kevent_create(device, channel);
if (event != NULL)
kevent_dispatch(&event->kevent);
2021-08-04 15:46:51 +02:00
}
/*
* This file is part of Ardix.
* Copyright (c) 2020, 2021 Felix Kopp <owo@fef.moe>.
*
* Ardix is non-violent software: you may only use, redistribute,
* and/or modify it under the terms of the CNPLv6+ as found in
* the LICENSE file in the source code root directory or at
* <https://git.pixie.town/thufie/CNPL>.
*
* Ardix comes with ABSOLUTELY NO WARRANTY, to the extent
* permitted by applicable law. See the CNPLv6+ for details.
*/