/**
@file
@brief kernel/userspace transport messages
@details Copyright (c) 2017-2020 Acronis International GmbH
@author Mikhail Krivtsov (mikhail.krivtsov@acronis.com)
@since $Id: $
*/
#pragma once
#include "file_contexts.h"
#include "si_common.h"
#include "subtype.h"
#include "thread_safe_path.h"
#include "task_info_map.h"
#include "transport.h"
#include "transport_protocol.h"
#include <linux/atomic.h>
#include <linux/list.h>
#include <linux/types.h> // bool, size_t, [u]int(8|16|32|64)_t
#include <linux/wait.h>
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
// 'msg_img_t' accessors
#define IMG_ID(msg_img) *(&(msg_img)->id)
#define IMG_REPLY(msg_img) *(&(msg_img)->reply)
#define IMG_TYPE(msg_img) *(&(msg_img)->type)
#define IMG_PAYLOAD(msg_img) (void *) (msg_img)->payload
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
typedef struct msg_s msg_t;
struct msg_s {
atomic_t ref_cnt;
atomic_t reply_wait_count; // number of expected replies
wait_queue_head_t wait_queue; // to wait for 'reply'
bool interrupted;
bool block;
// Can be 0, this means event does not require subtype filtering
SUBTYPE_MASK_TYPE subtype_mask;
thread_safe_path_t path;
thread_safe_path_t path2;
// This member store the additional necessary information for file context cache
task_info_t *task_info;
file_context_msg_info_t file_context_msg_info;
union
{
struct {
uint64_t pid_version;
} exec;
struct {
uint64_t pid_version;
} fork;
struct {
uint64_t pid_version;
int flags;
} open;
struct {
uint64_t pid_version;
int flags;
uint64_t low;
uint64_t high;
} write;
};
uint64_t id;
SiEvent event;
};
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
// 'msg_t' accessors
#define MSG_IMG(msg) ((request_msg_img_t*) (msg)->img)
#define MSG_SIZE(msg) (msg)->img_size
#define MSG_ID(msg) IMG_ID(MSG_IMG(msg))
#define MSG_REPLY(msg) IMG_REPLY(MSG_IMG(msg))
#define MSG_TYPE(msg) IMG_TYPE(MSG_IMG(msg))
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
typedef struct msg_sized_s msg_sized_t;
struct msg_sized_s {
size_t img_size;
uint8_t img[]; // 'request_msg_img_t'
};
// This should fit most of the messages other than the path ones
#define MSG_MAX_SMALL_PAYLOAD_SIZE (8 + sizeof(request_msg_img_t))
// Holds the varsized message either on stack or on heap based on the size.
typedef struct msg_varsized_s msg_varsized_t;
struct msg_varsized_s {
bool on_heap;
union {
struct {
msg_sized_t header; // header.img == img
uint8_t img[MSG_MAX_SMALL_PAYLOAD_SIZE];
} stack;
struct {
msg_sized_t* ptr;
} heap;
} data;
};
const char* msg_type_to_string(msg_type_t type);
const char* action_type_to_string(action_type_t type);
const char* return_type_to_string(return_type_t type);
msg_sized_t *msg_sized_new(size_t msg_img_size);
#define msg_sized_free mem_free
msg_t *msg_new(uint16_t operation, SUBTYPE_MASK_TYPE subtype_mask, uint16_t callback_type, uint64_t process_uid, size_t props_size);
msg_t *msg_new_with_alloc_flags(uint16_t operation, SUBTYPE_MASK_TYPE subtype_mask, uint16_t callback_type, uint64_t process_uid, size_t props_size, bool nowait);
msg_t *msg_ref(msg_t *msg);
void msg_unref(msg_t *msg);
msg_t *msg_reply_wait_count_inc(msg_t *msg);
void msg_reply_wait_count_dec(msg_t *msg);
msg_t *hello_msg_new(void);
msg_t *pong_msg_new(msg_sized_t *ping_msg, uint64_t event_uid);
int open_file_return_msg_new(msg_varsized_t *msg, int fd);
int version_info_return_msg_new(msg_varsized_t *msg);
int data_queue_offsets_return_msg_new(msg_varsized_t *msg, uint32_t size);
msg_t *heur_exec_msg_new(task_info_t *task_info, const transport_ids_t *ids);
int process_info_return_msg_new(msg_varsized_t *msg, pid_t nr);
int process_pid_version_return_msg_new(msg_varsized_t *msg, pid_t nr);
#define MSG_VARSIZED_GET_SIZED(msg) ((msg)->on_heap ? (msg)->data.heap.ptr : &(msg)->data.stack.header)
msg_sized_t* msg_varsized_init(msg_varsized_t *msg, size_t msg_img_size);
void msg_varsized_uninit(msg_varsized_t *msg);
|