81 lines
2.7 KiB
C
81 lines
2.7 KiB
C
/* SPDX-License-Identifier: GPL-2.0 */
|
|
/*
|
|
* Copyright (C) 2001, 2002 Jeff Dike (jdike@karaya.com)
|
|
*/
|
|
|
|
#ifndef __IRQ_KERN_H__
|
|
#define __IRQ_KERN_H__
|
|
|
|
#include <linux/interrupt.h>
|
|
#include <linux/time-internal.h>
|
|
#include <asm/ptrace.h>
|
|
#include "irq_user.h"
|
|
|
|
#define UM_IRQ_ALLOC -1
|
|
|
|
int um_request_irq(int irq, int fd, enum um_irq_type type,
|
|
irq_handler_t handler, unsigned long irqflags,
|
|
const char *devname, void *dev_id);
|
|
|
|
#ifdef CONFIG_UML_TIME_TRAVEL_SUPPORT
|
|
/**
|
|
* um_request_irq_tt - request an IRQ with timetravel handler
|
|
*
|
|
* @irq: the IRQ number, or %UM_IRQ_ALLOC
|
|
* @fd: The file descriptor to request an IRQ for
|
|
* @type: read or write
|
|
* @handler: the (generic style) IRQ handler
|
|
* @irqflags: Linux IRQ flags
|
|
* @devname: name for this to show
|
|
* @dev_id: data pointer to pass to the IRQ handler
|
|
* @timetravel_handler: the timetravel interrupt handler, invoked with the IRQ
|
|
* number, fd, dev_id and time-travel event pointer.
|
|
*
|
|
* Returns: The interrupt number assigned or a negative error.
|
|
*
|
|
* Note that the timetravel handler is invoked only if the time_travel_mode is
|
|
* %TT_MODE_EXTERNAL, and then it is invoked even while the system is suspended!
|
|
* This function must call time_travel_add_irq_event() for the event passed with
|
|
* an appropriate delay, before sending an ACK on the socket it was invoked for.
|
|
*
|
|
* If this was called while the system is suspended, then adding the event will
|
|
* cause the system to resume.
|
|
*
|
|
* Since this function will almost certainly have to handle the FD's condition,
|
|
* a read will consume the message, and after that it is up to the code using
|
|
* it to pass such a message to the @handler in whichever way it can.
|
|
*
|
|
* If time_travel_mode is not %TT_MODE_EXTERNAL the @timetravel_handler will
|
|
* not be invoked at all and the @handler must handle the FD becoming
|
|
* readable (or writable) instead. Use um_irq_timetravel_handler_used() to
|
|
* distinguish these cases.
|
|
*
|
|
* See virtio_uml.c for an example.
|
|
*/
|
|
int um_request_irq_tt(int irq, int fd, enum um_irq_type type,
|
|
irq_handler_t handler, unsigned long irqflags,
|
|
const char *devname, void *dev_id,
|
|
void (*timetravel_handler)(int, int, void *,
|
|
struct time_travel_event *));
|
|
#else
|
|
static inline
|
|
int um_request_irq_tt(int irq, int fd, enum um_irq_type type,
|
|
irq_handler_t handler, unsigned long irqflags,
|
|
const char *devname, void *dev_id,
|
|
void (*timetravel_handler)(int, int, void *,
|
|
struct time_travel_event *))
|
|
{
|
|
return um_request_irq(irq, fd, type, handler, irqflags,
|
|
devname, dev_id);
|
|
}
|
|
#endif
|
|
|
|
static inline bool um_irq_timetravel_handler_used(void)
|
|
{
|
|
return time_travel_mode == TT_MODE_EXTERNAL;
|
|
}
|
|
|
|
void um_free_irq(int irq, void *dev_id);
|
|
void free_irqs(void);
|
|
#endif
|