| /* |
| * SCLP |
| * Event Facility definitions |
| * |
| * Copyright IBM, Corp. 2012 |
| * |
| * Authors: |
| * Heinz Graalfs <graalfs@de.ibm.com> |
| * |
| * This work is licensed under the terms of the GNU GPL, version 2 or (at your |
| * option) any later version. See the COPYING file in the top-level directory. |
| * |
| */ |
| |
| #ifndef HW_S390_SCLP_EVENT_FACILITY_H |
| #define HW_S390_SCLP_EVENT_FACILITY_H |
| |
| #include "qemu/thread.h" |
| #include "hw/qdev-core.h" |
| #include "hw/s390x/sclp.h" |
| #include "qom/object.h" |
| |
| /* SCLP event types */ |
| #define SCLP_EVENT_OPRTNS_COMMAND 0x01 |
| #define SCLP_EVENT_MESSAGE 0x02 |
| #define SCLP_EVENT_CONFIG_MGT_DATA 0x04 |
| #define SCLP_EVENT_PMSGCMD 0x09 |
| #define SCLP_EVENT_ASCII_CONSOLE_DATA 0x1a |
| #define SCLP_EVENT_SIGNAL_QUIESCE 0x1d |
| |
| /* SCLP event masks */ |
| #define SCLP_EVMASK(T) (1ULL << (sizeof(sccb_mask_t) * 8 - (T))) |
| |
| #define SCLP_EVENT_MASK_OP_CMD SCLP_EVMASK(SCLP_EVENT_OPRTNS_COMMAND) |
| #define SCLP_EVENT_MASK_MSG SCLP_EVMASK(SCLP_EVENT_MESSAGE) |
| #define SCLP_EVENT_MASK_CONFIG_MGT_DATA SCLP_EVMASK(SCLP_EVENT_CONFIG_MGT_DATA) |
| #define SCLP_EVENT_MASK_PMSGCMD SCLP_EVMASK(SCLP_EVENT_PMSGCMD) |
| #define SCLP_EVENT_MASK_MSG_ASCII SCLP_EVMASK(SCLP_EVENT_ASCII_CONSOLE_DATA) |
| #define SCLP_EVENT_MASK_SIGNAL_QUIESCE SCLP_EVMASK(SCLP_EVENT_SIGNAL_QUIESCE) |
| |
| #define SCLP_UNCONDITIONAL_READ 0x00 |
| #define SCLP_SELECTIVE_READ 0x01 |
| |
| #define TYPE_SCLP_EVENT "s390-sclp-event-type" |
| OBJECT_DECLARE_TYPE(SCLPEvent, SCLPEventClass, |
| SCLP_EVENT) |
| |
| #define TYPE_SCLP_CPU_HOTPLUG "sclp-cpu-hotplug" |
| #define TYPE_SCLP_QUIESCE "sclpquiesce" |
| |
| #define SCLP_EVENT_MASK_LEN_MAX 1021 |
| |
| typedef struct WriteEventMask { |
| SCCBHeader h; |
| uint16_t _reserved; |
| uint16_t mask_length; |
| uint8_t masks[]; |
| /* |
| * Layout of the masks is |
| * uint8_t cp_receive_mask[mask_length]; |
| * uint8_t cp_send_mask[mask_length]; |
| * uint8_t receive_mask[mask_length]; |
| * uint8_t send_mask[mask_length]; |
| * where 1 <= mask_length <= SCLP_EVENT_MASK_LEN_MAX |
| */ |
| } QEMU_PACKED WriteEventMask; |
| |
| #define WEM_CP_RECEIVE_MASK(wem, mask_len) ((wem)->masks) |
| #define WEM_CP_SEND_MASK(wem, mask_len) ((wem)->masks + (mask_len)) |
| #define WEM_RECEIVE_MASK(wem, mask_len) ((wem)->masks + 2 * (mask_len)) |
| #define WEM_SEND_MASK(wem, mask_len) ((wem)->masks + 3 * (mask_len)) |
| |
| typedef uint64_t sccb_mask_t; |
| |
| typedef struct EventBufferHeader { |
| uint16_t length; |
| uint8_t type; |
| uint8_t flags; |
| uint16_t _reserved; |
| } QEMU_PACKED EventBufferHeader; |
| |
| typedef struct MdbHeader { |
| uint16_t length; |
| uint16_t type; |
| uint32_t tag; |
| uint32_t revision_code; |
| } QEMU_PACKED MdbHeader; |
| |
| typedef struct MTO { |
| uint16_t line_type_flags; |
| uint8_t alarm_control; |
| uint8_t _reserved[3]; |
| char message[]; |
| } QEMU_PACKED MTO; |
| |
| typedef struct GO { |
| uint32_t domid; |
| uint8_t hhmmss_time[8]; |
| uint8_t th_time[3]; |
| uint8_t _reserved_0; |
| uint8_t dddyyyy_date[7]; |
| uint8_t _reserved_1; |
| uint16_t general_msg_flags; |
| uint8_t _reserved_2[10]; |
| uint8_t originating_system_name[8]; |
| uint8_t job_guest_name[8]; |
| } QEMU_PACKED GO; |
| |
| #define MESSAGE_TEXT 0x0004 |
| |
| typedef struct MDBO { |
| uint16_t length; |
| uint16_t type; |
| union { |
| GO go; |
| MTO mto; |
| }; |
| } QEMU_PACKED MDBO; |
| |
| typedef struct MDB { |
| MdbHeader header; |
| MDBO mdbo[]; |
| } QEMU_PACKED MDB; |
| |
| typedef struct SclpMsg { |
| EventBufferHeader header; |
| MDB mdb; |
| } QEMU_PACKED SclpMsg; |
| |
| #define GDS_ID_MDSMU 0x1310 |
| #define GDS_ID_CPMSU 0x1212 |
| #define GDS_ID_TEXTCMD 0x1320 |
| |
| typedef struct GdsVector { |
| uint16_t length; |
| uint16_t gds_id; |
| } QEMU_PACKED GdsVector; |
| |
| #define GDS_KEY_SELFDEFTEXTMSG 0x31 |
| #define GDS_KEY_TEXTMSG 0x30 |
| |
| typedef struct GdsSubvector { |
| uint8_t length; |
| uint8_t key; |
| } QEMU_PACKED GdsSubvector; |
| |
| /* MDS Message Unit */ |
| typedef struct MDMSU { |
| GdsVector mdmsu; |
| GdsVector cpmsu; |
| GdsVector text_command; |
| GdsSubvector self_def_text_message; |
| GdsSubvector text_message; |
| } QEMU_PACKED MDMSU; |
| |
| typedef struct WriteEventData { |
| SCCBHeader h; |
| EventBufferHeader ebh; |
| } QEMU_PACKED WriteEventData; |
| |
| typedef struct ReadEventData { |
| SCCBHeader h; |
| union { |
| sccb_mask_t mask; |
| EventBufferHeader ebh; |
| }; |
| } QEMU_PACKED ReadEventData; |
| |
| struct SCLPEvent { |
| DeviceState qdev; |
| bool event_pending; |
| char *name; |
| }; |
| |
| struct SCLPEventClass { |
| DeviceClass parent_class; |
| int (*init)(SCLPEvent *event); |
| |
| /* get SCLP's send mask */ |
| sccb_mask_t (*get_send_mask)(void); |
| |
| /* get SCLP's receive mask */ |
| sccb_mask_t (*get_receive_mask)(void); |
| |
| int (*read_event_data)(SCLPEvent *event, EventBufferHeader *evt_buf_hdr, |
| int *slen); |
| |
| int (*write_event_data)(SCLPEvent *event, EventBufferHeader *evt_buf_hdr); |
| |
| /* can we handle this event type? */ |
| bool (*can_handle_event)(uint8_t type); |
| }; |
| |
| #define TYPE_SCLP_EVENT_FACILITY "s390-sclp-event-facility" |
| typedef struct SCLPEventFacility SCLPEventFacility; |
| typedef struct SCLPEventFacilityClass SCLPEventFacilityClass; |
| DECLARE_OBJ_CHECKERS(SCLPEventFacility, SCLPEventFacilityClass, |
| EVENT_FACILITY, TYPE_SCLP_EVENT_FACILITY) |
| |
| struct SCLPEventFacilityClass { |
| SysBusDeviceClass parent_class; |
| void (*command_handler)(SCLPEventFacility *ef, SCCB *sccb, uint64_t code); |
| bool (*event_pending)(SCLPEventFacility *ef); |
| }; |
| |
| BusState *sclp_get_event_facility_bus(void); |
| |
| #endif |