2013-11-27 16:28:31 +01:00
|
|
|
/*
|
2014-04-02 17:48:09 +02:00
|
|
|
* Copyright (C) 2014 Freie Universität Berlin
|
2011-12-01 16:27:23 +01:00
|
|
|
*
|
2014-08-23 15:43:13 +02:00
|
|
|
* 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.
|
2010-09-22 15:10:42 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2013-11-27 16:28:31 +01:00
|
|
|
* @defgroup core_msg Messaging / IPC
|
|
|
|
* @ingroup core
|
|
|
|
* @brief Messaging API for inter process communication
|
|
|
|
*
|
2014-04-01 15:45:06 +02:00
|
|
|
* 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.
|
2014-02-11 18:15:43 +01:00
|
|
|
*
|
2013-11-27 17:54:30 +01:00
|
|
|
* @{
|
2013-11-27 16:28:31 +01:00
|
|
|
*
|
|
|
|
* @file msg.h
|
|
|
|
* @brief Messaging API for inter process communication
|
|
|
|
*
|
2010-09-22 15:10:42 +02:00
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
2014-04-02 17:48:09 +02:00
|
|
|
* @author Kévin Roussel <Kevin.Roussel@inria.fr>
|
2010-09-22 15:10:42 +02:00
|
|
|
*/
|
|
|
|
|
2014-04-01 15:45:06 +02:00
|
|
|
#ifndef __MSG_H_
|
|
|
|
#define __MSG_H_
|
2010-09-22 15:10:42 +02:00
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdbool.h>
|
2014-07-06 22:57:56 +02:00
|
|
|
#include "kernel_types.h"
|
2010-09-22 15:10:42 +02:00
|
|
|
|
2014-10-13 14:44:28 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2010-09-22 15:10:42 +02:00
|
|
|
/**
|
|
|
|
* @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 {
|
2014-07-06 22:57:56 +02:00
|
|
|
kernel_pid_t sender_pid; /**< PID of sending thread. Will be filled in
|
2014-04-01 15:45:06 +02:00
|
|
|
by msg_send. */
|
|
|
|
uint16_t type; /**< Type field. */
|
2010-09-22 15:10:42 +02:00
|
|
|
union {
|
2014-04-01 15:45:06 +02:00
|
|
|
char *ptr; /**< Pointer content field. */
|
|
|
|
uint32_t value; /**< Value content field. */
|
|
|
|
} content; /**< Content of the message. */
|
2011-03-08 10:54:40 +01:00
|
|
|
} msg_t;
|
2010-09-22 15:10:42 +02:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
2014-09-02 18:32:37 +02:00
|
|
|
* @brief Send a message. (blocking)
|
2010-09-22 15:10:42 +02:00
|
|
|
*
|
2014-04-01 15:45:06 +02:00
|
|
|
* 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
|
2014-09-02 18:32:37 +02:00
|
|
|
*
|
|
|
|
* @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
|
2014-04-01 15:45:06 +02:00
|
|
|
*
|
|
|
|
* @return 1, if sending was successful (message delivered directly or to a
|
|
|
|
* queue)
|
2014-09-02 18:32:37 +02:00
|
|
|
* @return 0, if receiver is not waiting or has a full message queue
|
2014-04-01 15:45:06 +02:00
|
|
|
* @return -1, on error (invalid PID)
|
2010-09-22 15:10:42 +02:00
|
|
|
*/
|
2014-09-02 18:32:37 +02:00
|
|
|
int msg_try_send(msg_t *m, kernel_pid_t target_pid);
|
2010-09-22 15:10:42 +02:00
|
|
|
|
|
|
|
|
2014-04-02 17:48:09 +02:00
|
|
|
/**
|
|
|
|
* @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);
|
|
|
|
|
2010-09-22 15:10:42 +02:00
|
|
|
/**
|
|
|
|
* @brief Send message from interrupt.
|
|
|
|
*
|
2014-08-05 17:08:53 +02:00
|
|
|
* Will be automatically chosen instead of ``msg_send()`` if called from an
|
2014-04-01 15:45:06 +02:00
|
|
|
* interrupt/ISR.
|
2010-09-22 15:10:42 +02:00
|
|
|
*
|
2014-04-01 15:45:06 +02:00
|
|
|
* @param[in] m Pointer to preallocated ``msg_t`` structure, must
|
|
|
|
* not be NULL.
|
|
|
|
* @param[in] target_pid PID of target thread.
|
2010-09-22 15:10:42 +02:00
|
|
|
*
|
2014-04-01 15:45:06 +02:00
|
|
|
* @return 1, if sending was successful
|
|
|
|
* @return 0, if receiver is not waiting and ``block == 0``
|
2014-06-05 17:25:53 +02:00
|
|
|
* @return -1, on error (invalid PID)
|
2010-09-22 15:10:42 +02:00
|
|
|
*/
|
2014-07-06 22:57:56 +02:00
|
|
|
int msg_send_int(msg_t *m, kernel_pid_t target_pid);
|
2010-09-22 15:10:42 +02:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Receive a message.
|
|
|
|
*
|
|
|
|
* This function blocks until a message was received.
|
|
|
|
*
|
2014-04-01 15:45:06 +02:00
|
|
|
* @param[out] m Pointer to preallocated ``msg_t`` structure, must not be
|
|
|
|
* NULL.
|
|
|
|
*
|
|
|
|
* @return 1, Function always succeeds or blocks forever.
|
2010-09-22 15:10:42 +02:00
|
|
|
*/
|
2013-06-20 18:18:29 +02:00
|
|
|
int msg_receive(msg_t *m);
|
2010-09-22 15:10:42 +02:00
|
|
|
|
2013-07-05 19:22:29 +02:00
|
|
|
/**
|
|
|
|
* @brief Try to receive a message.
|
|
|
|
*
|
|
|
|
* This function does not block if no message can be received.
|
|
|
|
*
|
2014-04-01 15:45:06 +02:00
|
|
|
* @param[out] m Pointer to preallocated ``msg_t`` structure, must not be
|
|
|
|
* NULL.
|
|
|
|
*
|
|
|
|
* @return 1, if a message was received
|
|
|
|
* @return -1, otherwise.
|
2013-07-05 19:22:29 +02:00
|
|
|
*/
|
|
|
|
int msg_try_receive(msg_t *m);
|
|
|
|
|
2010-09-22 15:10:42 +02:00
|
|
|
/**
|
|
|
|
* @brief Send a message, block until reply received.
|
|
|
|
*
|
2014-04-01 15:45:06 +02:00
|
|
|
* 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.
|
|
|
|
* @param[in] target_pid The PID of the target process
|
|
|
|
*
|
|
|
|
* @return 1, if successful.
|
2010-09-22 15:10:42 +02:00
|
|
|
*/
|
2014-07-06 22:57:56 +02:00
|
|
|
int msg_send_receive(msg_t *m, msg_t *reply, kernel_pid_t target_pid);
|
2010-09-22 15:10:42 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Replies to a message.
|
|
|
|
*
|
|
|
|
* Sender must have sent the message with msg_send_receive().
|
|
|
|
*
|
2014-04-01 15:45:06 +02:00
|
|
|
* @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.
|
2010-09-22 15:10:42 +02:00
|
|
|
*
|
2014-04-01 15:45:06 +02:00
|
|
|
* @return 1, if successful
|
|
|
|
* @return 0, on error
|
2010-09-22 15:10:42 +02:00
|
|
|
*/
|
2011-03-08 10:54:40 +01:00
|
|
|
int msg_reply(msg_t *m, msg_t *reply);
|
2010-09-22 15:10:42 +02:00
|
|
|
|
2010-11-26 14:21:48 +01:00
|
|
|
/**
|
|
|
|
* @brief Initialize the current thread's message queue.
|
|
|
|
*
|
2014-04-01 15:45:06 +02:00
|
|
|
* @param[in] array Pointer to preallocated array of ``msg_t`` structures, must
|
|
|
|
* not be NULL.
|
2014-08-05 17:08:53 +02:00
|
|
|
* @param[in] num Number of ``msg_t`` structures in array.
|
2014-04-01 15:45:06 +02:00
|
|
|
* **MUST BE POWER OF TWO!**
|
2013-10-28 12:55:29 +01:00
|
|
|
*
|
2014-04-01 15:45:06 +02:00
|
|
|
* @return 0, if successful
|
|
|
|
* @return -1, on error
|
2010-11-26 14:21:48 +01:00
|
|
|
*/
|
2013-06-20 18:18:29 +02:00
|
|
|
int msg_init_queue(msg_t *array, int num);
|
2010-09-22 15:10:42 +02:00
|
|
|
|
2014-10-09 01:18:16 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2014-04-01 15:45:06 +02:00
|
|
|
#endif /* __MSG_H_ */
|
2010-09-22 15:10:42 +02:00
|
|
|
/** @} */
|