291 lines
9.8 KiB
C
291 lines
9.8 KiB
C
/* SPDX-License-Identifier: BSD-3-Clause */
|
|
/*
|
|
* Copyright (C) 2019 - 2023 Intel Corporation
|
|
*/
|
|
#ifndef _UAPI_LINUX_UM_TIMETRAVEL_H
|
|
#define _UAPI_LINUX_UM_TIMETRAVEL_H
|
|
#include <linux/types.h>
|
|
|
|
/**
|
|
* struct um_timetravel_msg - UM time travel message
|
|
*
|
|
* This is the basic message type, going in both directions.
|
|
*
|
|
* This is the message passed between the host (user-mode Linux instance)
|
|
* and the calendar (the application on the other side of the socket) in
|
|
* order to implement common scheduling.
|
|
*
|
|
* Whenever UML has an event it will request runtime for it from the
|
|
* calendar, and then wait for its turn until it can run, etc. Note
|
|
* that it will only ever request the single next runtime, i.e. multiple
|
|
* REQUEST messages override each other.
|
|
*/
|
|
struct um_timetravel_msg {
|
|
/**
|
|
* @op: operation value from &enum um_timetravel_ops
|
|
*/
|
|
__u32 op;
|
|
|
|
/**
|
|
* @seq: sequence number for the message - shall be reflected in
|
|
* the ACK response, and should be checked while processing
|
|
* the response to see if it matches
|
|
*/
|
|
__u32 seq;
|
|
|
|
/**
|
|
* @time: time in nanoseconds
|
|
*/
|
|
__u64 time;
|
|
};
|
|
|
|
/* max number of file descriptors that can be sent/received in a message */
|
|
#define UM_TIMETRAVEL_MAX_FDS 2
|
|
|
|
/**
|
|
* enum um_timetravel_shared_mem_fds - fds sent in ACK message for START message
|
|
*/
|
|
enum um_timetravel_shared_mem_fds {
|
|
/**
|
|
* @UM_TIMETRAVEL_SHARED_MEMFD: Index of the shared memory file
|
|
* descriptor in the control message
|
|
*/
|
|
UM_TIMETRAVEL_SHARED_MEMFD,
|
|
/**
|
|
* @UM_TIMETRAVEL_SHARED_LOGFD: Index of the logging file descriptor
|
|
* in the control message
|
|
*/
|
|
UM_TIMETRAVEL_SHARED_LOGFD,
|
|
UM_TIMETRAVEL_SHARED_MAX_FDS,
|
|
};
|
|
|
|
/**
|
|
* enum um_timetravel_start_ack - ack-time mask for start message
|
|
*/
|
|
enum um_timetravel_start_ack {
|
|
/**
|
|
* @UM_TIMETRAVEL_START_ACK_ID: client ID that controller allocated.
|
|
*/
|
|
UM_TIMETRAVEL_START_ACK_ID = 0xffff,
|
|
};
|
|
|
|
/**
|
|
* enum um_timetravel_ops - Operation codes
|
|
*/
|
|
enum um_timetravel_ops {
|
|
/**
|
|
* @UM_TIMETRAVEL_ACK: response (ACK) to any previous message,
|
|
* this usually doesn't carry any data in the 'time' field
|
|
* unless otherwise specified below, note: while using shared
|
|
* memory no ACK for WAIT and RUN messages, for more info see
|
|
* &struct um_timetravel_schedshm.
|
|
*/
|
|
UM_TIMETRAVEL_ACK = 0,
|
|
|
|
/**
|
|
* @UM_TIMETRAVEL_START: initialize the connection, the time
|
|
* field contains an (arbitrary) ID to possibly be able
|
|
* to distinguish the connections.
|
|
*/
|
|
UM_TIMETRAVEL_START = 1,
|
|
|
|
/**
|
|
* @UM_TIMETRAVEL_REQUEST: request to run at the given time
|
|
* (host -> calendar)
|
|
*/
|
|
UM_TIMETRAVEL_REQUEST = 2,
|
|
|
|
/**
|
|
* @UM_TIMETRAVEL_WAIT: Indicate waiting for the previously requested
|
|
* runtime, new requests may be made while waiting (e.g. due to
|
|
* interrupts); the time field is ignored. The calendar must process
|
|
* this message and later send a %UM_TIMETRAVEL_RUN message when
|
|
* the host can run again.
|
|
* (host -> calendar)
|
|
*/
|
|
UM_TIMETRAVEL_WAIT = 3,
|
|
|
|
/**
|
|
* @UM_TIMETRAVEL_GET: return the current time from the calendar in the
|
|
* ACK message, the time in the request message is ignored
|
|
* (host -> calendar)
|
|
*/
|
|
UM_TIMETRAVEL_GET = 4,
|
|
|
|
/**
|
|
* @UM_TIMETRAVEL_UPDATE: time update to the calendar, must be sent e.g.
|
|
* before kicking an interrupt to another calendar
|
|
* (host -> calendar)
|
|
*/
|
|
UM_TIMETRAVEL_UPDATE = 5,
|
|
|
|
/**
|
|
* @UM_TIMETRAVEL_RUN: run time request granted, current time is in
|
|
* the time field
|
|
* (calendar -> host)
|
|
*/
|
|
UM_TIMETRAVEL_RUN = 6,
|
|
|
|
/**
|
|
* @UM_TIMETRAVEL_FREE_UNTIL: Enable free-running until the given time,
|
|
* this is a message from the calendar telling the host that it can
|
|
* freely do its own scheduling for anything before the indicated
|
|
* time.
|
|
* Note that if a calendar sends this message once, the host may
|
|
* assume that it will also do so in the future, if it implements
|
|
* wraparound semantics for the time field.
|
|
* (calendar -> host)
|
|
*/
|
|
UM_TIMETRAVEL_FREE_UNTIL = 7,
|
|
|
|
/**
|
|
* @UM_TIMETRAVEL_GET_TOD: Return time of day, typically used once at
|
|
* boot by the virtual machines to get a synchronized time from
|
|
* the simulation.
|
|
*/
|
|
UM_TIMETRAVEL_GET_TOD = 8,
|
|
|
|
/**
|
|
* @UM_TIMETRAVEL_BROADCAST: Send/Receive a broadcast message.
|
|
* This message can be used to sync all components in the system
|
|
* with a single message, if the calender gets the message, the
|
|
* calender broadcast the message to all components, and if a
|
|
* component receives it it should act based on it e.g print a
|
|
* message to it's log system.
|
|
* (calendar <-> host)
|
|
*/
|
|
UM_TIMETRAVEL_BROADCAST = 9,
|
|
};
|
|
|
|
/* version of struct um_timetravel_schedshm */
|
|
#define UM_TIMETRAVEL_SCHEDSHM_VERSION 2
|
|
|
|
/**
|
|
* enum um_timetravel_schedshm_cap - time travel capabilities of every client
|
|
*
|
|
* These flags must be set immediately after processing the ACK to
|
|
* the START message, before sending any message to the controller.
|
|
*/
|
|
enum um_timetravel_schedshm_cap {
|
|
/**
|
|
* @UM_TIMETRAVEL_SCHEDSHM_CAP_TIME_SHARE: client can read current time
|
|
* update internal time request to shared memory and read
|
|
* free until and send no Ack on RUN and doesn't expect ACK on
|
|
* WAIT.
|
|
*/
|
|
UM_TIMETRAVEL_SCHEDSHM_CAP_TIME_SHARE = 0x1,
|
|
};
|
|
|
|
/**
|
|
* enum um_timetravel_schedshm_flags - time travel flags of every client
|
|
*/
|
|
enum um_timetravel_schedshm_flags {
|
|
/**
|
|
* @UM_TIMETRAVEL_SCHEDSHM_FLAGS_REQ_RUN: client has a request to run.
|
|
* It's set by client when it has a request to run, if (and only
|
|
* if) the @running_id points to a client that is able to use
|
|
* shared memory, i.e. has %UM_TIMETRAVEL_SCHEDSHM_CAP_TIME_SHARE
|
|
* (this includes the client itself). Otherwise, a message must
|
|
* be used.
|
|
*/
|
|
UM_TIMETRAVEL_SCHEDSHM_FLAGS_REQ_RUN = 0x1,
|
|
};
|
|
|
|
/**
|
|
* DOC: Time travel shared memory overview
|
|
*
|
|
* The main purpose of the shared memory is to avoid all time travel message
|
|
* that don't need any action, for example current time can be held in shared
|
|
* memory without the need of any client to send a message UM_TIMETRAVEL_GET
|
|
* in order to know what's the time.
|
|
*
|
|
* Since this is shared memory with all clients and controller and controller
|
|
* creates the shared memory space, all time values are absolute to controller
|
|
* time. So first time client connects to shared memory mode it should take the
|
|
* current_time value in shared memory and keep it internally as a diff to
|
|
* shared memory times, and once shared memory is initialized, any interaction
|
|
* with the controller must happen in the controller time domain, including any
|
|
* messages (for clients that are not using shared memory, the controller will
|
|
* handle an offset and make the clients think they start at time zero.)
|
|
*
|
|
* Along with the shared memory file descriptor is sent to the client a logging
|
|
* file descriptor, to have all logs related to shared memory,
|
|
* logged into one place. note: to have all logs synced into log file at write,
|
|
* file should be flushed (fflush) after writing to it.
|
|
*
|
|
* To avoid memory corruption, we define below for each field who can write to
|
|
* it at what time, defined in the structure fields.
|
|
*
|
|
* To avoid having to pack this struct, all fields in it must be naturally aligned
|
|
* (i.e. aligned to their size).
|
|
*/
|
|
|
|
/**
|
|
* union um_timetravel_schedshm_client - UM time travel client struct
|
|
*
|
|
* Every entity using the shared memory including the controller has a place in
|
|
* the um_timetravel_schedshm clients array, that holds info related to the client
|
|
* using the shared memory, and can be set only by the client after it gets the
|
|
* fd memory.
|
|
*
|
|
* @capa: bit fields with client capabilities see
|
|
* &enum um_timetravel_schedshm_cap, set by client once after getting the
|
|
* shared memory file descriptor.
|
|
* @flags: bit fields for flags see &enum um_timetravel_schedshm_flags for doc.
|
|
* @req_time: request time to run, set by client on every request it needs.
|
|
* @name: unique id sent to the controller by client with START message.
|
|
*/
|
|
union um_timetravel_schedshm_client {
|
|
struct {
|
|
__u32 capa;
|
|
__u32 flags;
|
|
__u64 req_time;
|
|
__u64 name;
|
|
};
|
|
char reserve[128]; /* reserved for future usage */
|
|
};
|
|
|
|
/**
|
|
* struct um_timetravel_schedshm - UM time travel shared memory struct
|
|
*
|
|
* @hdr: header fields:
|
|
* @version: Current version struct UM_TIMETRAVEL_SCHEDSHM_VERSION,
|
|
* set by controller once at init, clients must check this after mapping
|
|
* and work without shared memory if they cannot handle the indicated
|
|
* version.
|
|
* @len: Length of all the memory including header (@hdr), clients should once
|
|
* per connection first mmap the header and take the length (@len) to remap the entire size.
|
|
* This is done in order to support dynamic struct size letting number of
|
|
* clients be dynamic based on controller support.
|
|
* @free_until: Stores the next request to run by any client, in order for the
|
|
* current client to know how long it can still run. A client needs to (at
|
|
* least) reload this value immediately after communicating with any other
|
|
* client, since the controller will update this field when a new request
|
|
* is made by any client. Clients also must update this value when they
|
|
* insert/update an own request into the shared memory while not running
|
|
* themselves, and the new request is before than the current value.
|
|
* current_time: Current time, can only be set by the client in running state
|
|
* (indicated by @running_id), though that client may only run until @free_until,
|
|
* so it must remain smaller than @free_until.
|
|
* @running_id: The current client in state running, set before a client is
|
|
* notified that it's now running.
|
|
* @max_clients: size of @clients array, set once at init by the controller.
|
|
* @clients: clients array see &union um_timetravel_schedshm_client for doc,
|
|
* set only by client.
|
|
*/
|
|
struct um_timetravel_schedshm {
|
|
union {
|
|
struct {
|
|
__u32 version;
|
|
__u32 len;
|
|
__u64 free_until;
|
|
__u64 current_time;
|
|
__u16 running_id;
|
|
__u16 max_clients;
|
|
};
|
|
char hdr[4096]; /* align to 4K page size */
|
|
};
|
|
union um_timetravel_schedshm_client clients[];
|
|
};
|
|
#endif /* _UAPI_LINUX_UM_TIMETRAVEL_H */
|