⛏️ index : haiku.git

/*
 * Copyright 2005-2008, Haiku Inc. All Rights Reserved.
 * Distributed under the terms of the MIT License.
 */
#ifndef _KERNEL_EXPORT_H
#define _KERNEL_EXPORT_H


#include <SupportDefs.h>
#include <OS.h>


/* interrupts and spinlocks */

typedef ulong cpu_status;

// WARNING: For Haiku debugging only! This changes the spinlock type in a
// binary incompatible way!
//#define B_DEBUG_SPINLOCK_CONTENTION	1

#if B_DEBUG_SPINLOCK_CONTENTION
	typedef struct {
		int32	lock;
		int32		failed_try_acquire;
		bigtime_t	total_wait;
		bigtime_t	total_held;
		bigtime_t	last_acquired;
	} spinlock;

#	define B_SPINLOCK_INITIALIZER { 0, 0, 0, 0, 0 }
#	define B_INITIALIZE_SPINLOCK(spinlock)	do {	\
			(spinlock)->lock = 0;					\
			(spinlock)->failed_try_acquire = 0;		\
			(spinlock)->total_wait = 0;				\
			(spinlock)->total_held = 0;				\
			(spinlock)->last_acquired = 0;			\
		} while (false)
#else
	typedef struct {
		int32	lock;
	} spinlock;

#	define B_SPINLOCK_INITIALIZER { 0 }
#	define B_INITIALIZE_SPINLOCK(spinlock)	do {	\
			(spinlock)->lock = 0;					\
		} while (false)
#endif

#define B_SPINLOCK_IS_LOCKED(spinlock)	(atomic_get(&(spinlock)->lock) > 0)

typedef struct {
	int32		lock;
} rw_spinlock;

#define B_RW_SPINLOCK_INITIALIZER	{ 0 }
#define B_INITIALIZE_RW_SPINLOCK(rw_spinlock)	do {	\
		(rw_spinlock)->lock = 0;						\
	} while (false)

typedef struct {
	spinlock	lock;
	uint32		count;
} seqlock;

#define B_SEQLOCK_INITIALIZER	{ B_SPINLOCK_INITIALIZER, 0 }
#define B_INITIALIZE_SEQLOCK(seqlock)	do {		\
		B_INITIALIZE_SPINLOCK(&(seqlock)->lock);	\
		(seqlock)->count = 0;						\
	} while (false)

/* interrupt handling support for device drivers */

typedef int32 (*interrupt_handler)(void *data);

/* Values returned by interrupt handlers */
#define B_UNHANDLED_INTERRUPT	0	/* pass to next handler */
#define B_HANDLED_INTERRUPT		1	/* don't pass on */
#define B_INVOKE_SCHEDULER		2	/* don't pass on; invoke the scheduler */

/* Flags that can be passed to install_io_interrupt_handler() */
#define B_NO_ENABLE_COUNTER		1


/* timer interrupts services */

typedef struct timer timer;
typedef	int32 (*timer_hook)(timer *);

struct timer {
	struct timer *next;
	int64		schedule_time;
	void		*user_data;
	uint16		flags;
	uint16		cpu;
	timer_hook	hook;
	bigtime_t	period;
};

#define B_ONE_SHOT_ABSOLUTE_TIMER	1
#define	B_ONE_SHOT_RELATIVE_TIMER	2
#define	B_PERIODIC_TIMER			3


/* virtual memory buffer functions */

#define B_DMA_IO		0x00000001
#define B_READ_DEVICE	0x00000002

typedef struct {
	phys_addr_t	address;	/* address in physical memory */
	phys_size_t	size;		/* size of block */
} physical_entry;

/* address specifications for mapping physical memory */
#define	B_ANY_KERNEL_BLOCK_ADDRESS	(B_ANY_KERNEL_ADDRESS + 1)

/* memory types for physical memory */
#define B_UNCACHED_MEMORY			(1 << 28)
#define B_WRITE_COMBINING_MEMORY	(2 << 28)
#define B_WRITE_THROUGH_MEMORY		(3 << 28)
#define B_WRITE_PROTECTED_MEMORY	(4 << 28)
#define B_WRITE_BACK_MEMORY			(5 << 28)
#define B_MEMORY_TYPE_MASK			(0xf0000000)

/* area protection flags for the kernel */
#define B_KERNEL_READ_AREA			(1 << 4)
#define B_KERNEL_WRITE_AREA			(1 << 5)
#define B_KERNEL_EXECUTE_AREA		(1 << 6)
#define B_KERNEL_STACK_AREA			(1 << 7)


/* kernel daemon service */

typedef void (*daemon_hook)(void *arg, int iteration);


/* kernel debugging facilities */

/* special return codes for kernel debugger */
#define B_KDEBUG_CONT	2
#define B_KDEBUG_QUIT	3

typedef int (*debugger_command_hook)(int argc, char **argv);


#ifdef __cplusplus
extern "C" {
#endif

/* interrupts, spinlock, and timers */
extern cpu_status	disable_interrupts(void);
extern void			restore_interrupts(cpu_status status);

extern void			acquire_spinlock(spinlock *lock);
extern void			release_spinlock(spinlock *lock);

extern bool			try_acquire_write_spinlock(rw_spinlock* lock);
extern void			acquire_write_spinlock(rw_spinlock* lock);
extern void			release_write_spinlock(rw_spinlock* lock);
extern bool			try_acquire_read_spinlock(rw_spinlock* lock);
extern void			acquire_read_spinlock(rw_spinlock* lock);
extern void			release_read_spinlock(rw_spinlock* lock);

extern bool			try_acquire_write_seqlock(seqlock* lock);
extern void			acquire_write_seqlock(seqlock* lock);
extern void			release_write_seqlock(seqlock* lock);
extern uint32		acquire_read_seqlock(seqlock* lock);
extern bool			release_read_seqlock(seqlock* lock, uint32 count);

extern status_t		install_io_interrupt_handler(int32 interrupt_number,
						interrupt_handler handler, void *data, uint32 flags);
extern status_t		remove_io_interrupt_handler(int32 interrupt_number,
						interrupt_handler handler, void	*data);

extern status_t		add_timer(timer *t, timer_hook hook, bigtime_t period,
						int32 flags);
extern bool			cancel_timer(timer *t);

/* kernel threads */
extern thread_id	spawn_kernel_thread(thread_func function,
						const char *name, int32 priority, void *arg);

/* signal functions */
extern int			send_signal_etc(pid_t thread, uint signal, uint32 flags);

/* virtual memory */
extern status_t		lock_memory_etc(team_id team, void *buffer, size_t numBytes,
						uint32 flags);
extern status_t		lock_memory(void *buffer, size_t numBytes, uint32 flags);
extern status_t		unlock_memory_etc(team_id team, void *address,
						size_t numBytes, uint32 flags);
extern status_t		unlock_memory(void *buffer, size_t numBytes, uint32 flags);
extern status_t		get_memory_map_etc(team_id team, const void *address,
						size_t numBytes, physical_entry *table,
						uint32* _numEntries);
extern int32		get_memory_map(const void *buffer, size_t size,
						physical_entry *table, int32 numEntries);
extern area_id		map_physical_memory(const char *areaName,
						phys_addr_t physicalAddress, size_t size, uint32 flags,
						uint32 protection, void **_mappedAddress);

/* kernel debugging facilities */
#if defined(_KERNEL_MODE) || defined(_BOOT_MODE)
extern void			dprintf(const char *format, ...) _PRINTFLIKE(1, 2);
#endif
extern void			dvprintf(const char *format, va_list args);
extern void			kprintf(const char *fmt, ...) _PRINTFLIKE(1, 2);

extern void			dump_block(const char *buffer, int size, const char *prefix);
						/* TODO: temporary API: hexdumps given buffer */

extern bool			set_dprintf_enabled(bool new_state);

extern void			panic(const char *format, ...) _PRINTFLIKE(1, 2);

extern void			kernel_debugger(const char *message);
extern uint64		parse_expression(const char *string);

extern int			add_debugger_command(const char *name,
						debugger_command_hook hook, const char *help);
extern int			remove_debugger_command(const char *name,
						debugger_command_hook hook);

/* Miscellaneous */
extern void			spin(bigtime_t microseconds);

extern status_t		register_kernel_daemon(daemon_hook hook, void *arg,
						int frequency);
extern status_t		unregister_kernel_daemon(daemon_hook hook, void *arg);

extern void			call_all_cpus(void (*func)(void *, int), void *cookie);
extern void			call_all_cpus_sync(void (*func)(void *, int), void *cookie);
extern void			memory_read_barrier(void);
extern void			memory_write_barrier(void);

/* safe methods to access user memory without having to lock it */
extern status_t		user_memcpy(void *to, const void *from, size_t size);
extern ssize_t		user_strlcpy(char *to, const char *from, size_t size);
extern status_t		user_memset(void *start, char c, size_t count);

#ifdef __cplusplus
}
#endif

#endif	/* _KERNEL_EXPORT_H */