1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/core/include/msg.h

243 lines
7.4 KiB
C

/*
* Copyright (C) 2014 Freie Universität Berlin
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @defgroup core_msg Messaging / IPC
* @ingroup core
* @brief Messaging API for inter process communication
*
* There are two ways to use the IPC Messaging system of RIOT. The default is
* synchronous messaging. In this manner, messages are either dropped when the
* receiver is not waiting and the message was sent non-blocking, or will be
* delivered immediately when the receiver calls msg_receive(msg_t* m). To use
* asynchronous messaging any thread can create its own queue by calling
* msg_init_queue(msg_t* array, int num). Messages sent to a thread with a non
* full message queue are never dropped * and the sending never blocks. Threads
* with a full message queue behaves like in synchronous mode.
*
* @{
*
* @file
* @brief Messaging API for inter process communication
*
* @author Kaspar Schleiser <kaspar@schleiser.de>
* @author Kévin Roussel <Kevin.Roussel@inria.fr>
*/
#ifndef MSG_H_
#define MSG_H_
#include <stdint.h>
#include <stdbool.h>
#include "kernel_types.h"
#ifdef __cplusplus
extern "C" {
#endif
/**
* @brief Describes a message object which can be sent between threads.
*
* User can set type and one of content.ptr and content.value. (content is a union)
* The meaning of type and the content fields is totally up to the user,
* the corresponding fields are never read by the kernel.
*
*/
typedef struct msg {
kernel_pid_t sender_pid; /**< PID of sending thread. Will be filled in
by msg_send. */
uint16_t type; /**< Type field. */
union {
char *ptr; /**< Pointer content field. */
uint32_t value; /**< Value content field. */
} content; /**< Content of the message. */
} msg_t;
/**
* @brief Send a message (blocking).
*
* This function sends a message to another thread. The ``msg_t`` structure has
* to be allocated (e.g. on the stack) before calling the function and can be
* freed afterwards. If called from an interrupt, this function will never
* block.
*
* @param[in] m Pointer to preallocated ``msg_t`` structure, must
* not be NULL.
* @param[in] target_pid PID of target thread
*
* @return 1, if sending was successful (message delivered directly or to a
* queue)
* @return 0, if called from ISR and receiver cannot receive the message now
* (it is not waiting or it's message queue is full)
* @return -1, on error (invalid PID)
*/
int msg_send(msg_t *m, kernel_pid_t target_pid);
/**
* @brief Send a message (non-blocking).
*
* This function sends a message to another thread. The ``msg_t`` structure has
* to be allocated (e.g. on the stack) before calling the function and can be
* freed afterwards. This function will never block.
*
* @param[in] m Pointer to preallocated ``msg_t`` structure, must
* not be NULL.
* @param[in] target_pid PID of target thread
*
* @return 1, if sending was successful (message delivered directly or to a
* queue)
* @return 0, if receiver is not waiting or has a full message queue
* @return -1, on error (invalid PID)
*/
int msg_try_send(msg_t *m, kernel_pid_t target_pid);
/**
* @brief Send a message to the current thread.
* @details Will work only if the thread has a message queue.
*
* Will be automatically chosen instead of @c msg_send
* if @c target_pid == @c thread_pid.
* This function never blocks.
*
* @param m pointer to message structure
*
* @return 1 if sending was successful
* @return 0 if the thread's message queue is full (or inexistent)
*/
int msg_send_to_self(msg_t *m);
/**
* Value of msg_t::sender_pid if the sender was an interrupt service routine.
*/
#define KERNEL_PID_ISR (KERNEL_PID_LAST + 1)
/**
* @brief Send message from interrupt.
*
* Will be automatically chosen instead of msg_send() if called from an
* interrupt/ISR.
*
* The value of ``m->sender_pid`` is set to @ref KERNEL_PID_ISR.
*
* @see msg_sent_by_int()
*
* @param[in] m Pointer to preallocated @ref msg_t structure, must
* not be NULL.
* @param[in] target_pid PID of target thread.
*
* @return 1, if sending was successful
* @return 0, if receiver is not waiting and ``block == 0``
* @return -1, on error (invalid PID)
*/
int msg_send_int(msg_t *m, kernel_pid_t target_pid);
/**
* @brief Test if the message was sent inside an ISR.
* @see msg_send_int()
* @param[in] m The message in question.
* @returns `== 0` if *not* sent by an ISR
* @returns `!= 0` if sent by an ISR
*/
static inline int msg_sent_by_int(const msg_t *m)
{
return (m->sender_pid == KERNEL_PID_ISR);
}
/**
* @brief Receive a message.
*
* This function blocks until a message was received.
*
* @param[out] m Pointer to preallocated ``msg_t`` structure, must not be
* NULL.
*
* @return 1, Function always succeeds or blocks forever.
*/
int msg_receive(msg_t *m);
/**
* @brief Try to receive a message.
*
* This function does not block if no message can be received.
*
* @param[out] m Pointer to preallocated ``msg_t`` structure, must not be
* NULL.
*
* @return 1, if a message was received
* @return -1, otherwise.
*/
int msg_try_receive(msg_t *m);
/**
* @brief Send a message, block until reply received.
*
* This function sends a message to *target_pid* and then blocks until target
* has sent a reply which is then stored in *reply*.
*
* @note CAUTION! Use this function only when receiver is already waiting.
* If not use simple msg_send()
* @param[in] m Pointer to preallocated ``msg_t`` structure with
* the message to send, must not be NULL.
* @param[out] reply Pointer to preallocated msg. Reply will be written
* here, must not be NULL. Can be identical to @p m.
* @param[in] target_pid The PID of the target process
*
* @return 1, if successful.
*/
int msg_send_receive(msg_t *m, msg_t *reply, kernel_pid_t target_pid);
/**
* @brief Replies to a message.
*
* Sender must have sent the message with msg_send_receive().
*
* @param[in] m message to reply to, must not be NULL.
* @param[out] reply message that target will get as reply, must not be NULL.
*
* @return 1, if successful
* @return -1, on error
*/
int msg_reply(msg_t *m, msg_t *reply);
/**
* @brief Replies to a message from interrupt.
*
* An ISR can obviously not receive messages, however a thread might delegate
* replying to a message to an ISR.
*
* @param[in] m message to reply to, must not be NULL.
* @param[out] reply message that target will get as reply, must not be NULL.
*
* @return 1, if successful
* @return -1, on error
*/
int msg_reply_int(msg_t *m, msg_t *reply);
/**
* @brief Initialize the current thread's message queue.
*
* @param[in] array Pointer to preallocated array of ``msg_t`` structures, must
* not be NULL.
* @param[in] num Number of ``msg_t`` structures in array.
* **MUST BE POWER OF TWO!**
*
* @return 0, if successful
* @return -1, on error
*/
int msg_init_queue(msg_t *array, int num);
#ifdef __cplusplus
}
#endif
#endif /* MSG_H_ */
/** @} */