/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/

/***
  This file is part of systemd.

  Copyright (C) 2014 David Herrmann <dh.herrmann@gmail.com>

  systemd is free software; you can redistribute it and/or modify it
  under the terms of the GNU Lesser General Public License as published by
  the Free Software Foundation; either version 2.1 of the License, or
  (at your option) any later version.

  systemd is distributed in the hope that it will be useful, but
  WITHOUT ANY WARRANTY; without even the implied warranty of
  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  Lesser General Public License for more details.

  You should have received a copy of the GNU Lesser General Public License
  along with systemd; If not, see <http://www.gnu.org/licenses/>.
***/

#pragma once

#include <inttypes.h>
#include <libudev.h>
#include <linux/input.h>
#include <stdbool.h>
#include <stdlib.h>
#include <systemd/sd-bus.h>
#include <systemd/sd-event.h>
#include <xkbcommon/xkbcommon.h>
#include "hashmap.h"
#include "idev.h"
#include "list.h"
#include "util.h"

typedef struct idev_link                idev_link;
typedef struct idev_device_vtable       idev_device_vtable;
typedef struct idev_element             idev_element;
typedef struct idev_element_vtable      idev_element_vtable;

/*
 * Evdev Elements
 */

bool idev_is_evdev(idev_element *e);
idev_element *idev_find_evdev(idev_session *s, dev_t devnum);
int idev_evdev_new(idev_element **out, idev_session *s, struct udev_device *ud);

/*
 * Keyboard Devices
 */

bool idev_is_keyboard(idev_device *d);
idev_device *idev_find_keyboard(idev_session *s, const char *name);
int idev_keyboard_new(idev_device **out, idev_session *s, const char *name);

/*
 * Element Links
 */

struct idev_link {
        /* element-to-device connection */
        LIST_FIELDS(idev_link, links_by_element);
        idev_element *element;

        /* device-to-element connection */
        LIST_FIELDS(idev_link, links_by_device);
        idev_device *device;
};

/*
 * Devices
 */

struct idev_device_vtable {
        void (*free) (idev_device *d);
        void (*attach) (idev_device *d, idev_link *l);
        void (*detach) (idev_device *d, idev_link *l);
        int (*feed) (idev_device *d, idev_data *data);
};

struct idev_device {
        const idev_device_vtable *vtable;
        idev_session *session;
        char *name;

        LIST_HEAD(idev_link, links);

        bool public : 1;
        bool enabled : 1;
};

#define IDEV_DEVICE_INIT(_vtable, _session) ((idev_device){ \
                .vtable = (_vtable), \
                .session = (_session), \
        })

idev_device *idev_find_device(idev_session *s, const char *name);

int idev_device_add(idev_device *d, const char *name);
idev_device *idev_device_free(idev_device *d);

DEFINE_TRIVIAL_CLEANUP_FUNC(idev_device*, idev_device_free);

int idev_device_feed(idev_device *d, idev_data *data);
void idev_device_feedback(idev_device *d, idev_data *data);

/*
 * Elements
 */

struct idev_element_vtable {
        void (*free) (idev_element *e);
        void (*enable) (idev_element *e);
        void (*disable) (idev_element *e);
        void (*open) (idev_element *e);
        void (*close) (idev_element *e);
        void (*resume) (idev_element *e, int fd);
        void (*pause) (idev_element *e, const char *mode);
        void (*feedback) (idev_element *e, idev_data *data);
};

struct idev_element {
        const idev_element_vtable *vtable;
        idev_session *session;
        unsigned long n_open;
        char *name;

        LIST_HEAD(idev_link, links);

        bool enabled : 1;
        bool readable : 1;
        bool writable : 1;
};

#define IDEV_ELEMENT_INIT(_vtable, _session) ((idev_element){ \
                .vtable = (_vtable), \
                .session = (_session), \
        })

idev_element *idev_find_element(idev_session *s, const char *name);

int idev_element_add(idev_element *e, const char *name);
idev_element *idev_element_free(idev_element *e);

DEFINE_TRIVIAL_CLEANUP_FUNC(idev_element*, idev_element_free);

int idev_element_feed(idev_element *e, idev_data *data);
void idev_element_feedback(idev_element *e, idev_data *data);

/*
 * Sessions
 */

struct idev_session {
        idev_context *context;
        char *name;
        char *path;
        sd_bus_slot *slot_resume_device;
        sd_bus_slot *slot_pause_device;

        Hashmap *element_map;
        Hashmap *device_map;

        idev_event_fn event_fn;
        void *userdata;

        bool custom : 1;
        bool managed : 1;
        bool enabled : 1;
};

idev_session *idev_find_session(idev_context *c, const char *name);
int idev_session_raise_device_data(idev_session *s, idev_device *d, idev_data *data);

/*
 * Contexts
 */

struct idev_context {
        unsigned long ref;
        sd_event *event;
        sd_bus *sysbus;

        Hashmap *session_map;
        Hashmap *data_map;
};