acrn-hypervisor/misc/services/life_mngr/life_mngr_win.c

291 lines
8.3 KiB
C

/*
* Copyright (C)2019-2022 Intel Corporation.
* SPDX-License-Identifier: BSD-3-Clause
*/
#include <stdio.h>
#include <stdbool.h>
#include <winsock2.h>
#include <windows.h>
#define WIN_VM_NAME "windows"
#define REQ_SYS_SHUTDOWN "req_sys_shutdown"
#define ACK_REQ_SYS_SHUTDOWN "ack_req_sys_shutdown"
#define SYNC_CMD "sync"
#define ACK_SYNC "ack_sync"
#define POWEROFF_CMD "poweroff_cmd"
#define ACK_POWEROFF "ack_poweroff"
#define USER_VM_SHUTDOWN "user_vm_shutdown"
#define ACK_USER_VM_SHUTDOWN "ack_user_vm_shutdown"
#define USER_VM_REBOOT "user_vm_reboot"
#define ACK_USER_VM_REBOOT "ack_user_vm_reboot"
#define REQ_SYS_REBOOT "req_sys_reboot"
#define ACK_REQ_SYS_REBOOT "ack_req_sys_reboot"
#define SYNC_FMT "sync:%s"
#define S5_REJECTED "system shutdown request is rejected"
#define BUFF_SIZE (32U)
#define MSG_SIZE (8U)
#define UVM_SOCKET_PORT (0x2001U)
#define READ_INTERVAL (100U) /* The time unit is microsecond */
#define MIN_RESEND_TIME (3U)
#define SECOND_TO_MS (1000U)
#define RETRY_RECV_TIMES (100U)
HANDLE hCom2;
unsigned int resend_time;
char resend_buf[BUFF_SIZE];
void send_message_by_uart(HANDLE hCom, char *buf, unsigned int len)
{
int i;
DWORD written;
for (i = 0; i < len; i++)
WriteFile(hCom, &buf[i], 1, &written, NULL);
WriteFile(hCom, "\n", 1, &written, NULL);
}
void start_uart_resend(char *buf, unsigned int time)
{
if (resend_time < MIN_RESEND_TIME)
resend_time = MIN_RESEND_TIME;
strncpy(resend_buf, buf, BUFF_SIZE - 1);
resend_time = time + 1U;
}
void stop_uart_resend(void)
{
memset(resend_buf, 0x0, BUFF_SIZE);
resend_time = 0U;
}
void handle_socket_request(SOCKET sClient, char *req_message)
{
char ack_message[BUFF_SIZE];
snprintf(ack_message, sizeof(ack_message), "ack_%s", req_message);
/**
* The lifecycle manager in Service VM checks sync message every 5 seconds
* during listening phase, delay 6 seconds to wait Service VM to receive the
* sync message, then start to send message to Service VM.
*/
Sleep(6U * SECOND_TO_MS);
send(sClient, ack_message, sizeof(ack_message), 0);
start_uart_resend(req_message, MIN_RESEND_TIME);
send_message_by_uart(hCom2, req_message, strnlen(req_message, BUFF_SIZE));
Sleep(2U * READ_INTERVAL);
return;
}
DWORD WINAPI open_socket_server(LPVOID lpParam)
{
WORD sockVersion = MAKEWORD(2, 2);
WSADATA wsaData;
char revData[BUFF_SIZE];
struct sockaddr_in sin;
SOCKET sClient;
struct sockaddr_in remoteAddr;
int nAddrlen = sizeof(remoteAddr);
int ret;
ret = WSAStartup(sockVersion, &wsaData);
if (ret != 0) {
printf("Failed to initiate Windows Socket, error: %d\n", ret);
return -1;
}
SOCKET slisten = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (slisten == INVALID_SOCKET) {
printf("Socket error !\n");
goto start_error;
}
sin.sin_family = AF_INET;
sin.sin_port = htons(UVM_SOCKET_PORT);
sin.sin_addr.S_un.S_addr = INADDR_ANY;
if (bind(slisten, (LPSOCKADDR)&sin, sizeof(sin)) == SOCKET_ERROR) {
printf("Bind error !\n");
goto sock_exit;
}
if (listen(slisten, 5) == SOCKET_ERROR) {
printf("Listen error !\n");
goto sock_exit;
}
printf("Wait for connect ...\n");
sClient = accept(slisten, (SOCKADDR *)&remoteAddr, &nAddrlen);
if (sClient == INVALID_SOCKET) {
printf("Accept error\n");
goto sock_exit;
}
printf("Accept one connect %s\n", inet_ntoa(remoteAddr.sin_addr));
do {
memset(revData, 0, sizeof(revData));
int ret = recv(sClient, revData, BUFF_SIZE, 0);
if (ret > 0) {
revData[ret] = 0x00;
printf(revData);
}
Sleep(READ_INTERVAL);
if (strncmp(revData, REQ_SYS_SHUTDOWN, sizeof(REQ_SYS_SHUTDOWN)) == 0) {
handle_socket_request(sClient, REQ_SYS_SHUTDOWN);
break;
}
if (strncmp(revData, REQ_SYS_REBOOT, sizeof(REQ_SYS_REBOOT)) == 0) {
handle_socket_request(sClient, REQ_SYS_REBOOT);
break;
}
} while (1);
closesocket(sClient);
sock_exit:
closesocket(slisten);
start_error:
WSACleanup();
return 0;
}
HANDLE initCom(const char *szStr)
{
HANDLE hCom = CreateFile(szStr,
GENERIC_READ | GENERIC_WRITE,
0,
NULL,
OPEN_EXISTING,
0,
NULL);
if (hCom == INVALID_HANDLE_VALUE)
{
printf("Opening %s failed!\n", szStr);
return hCom;
}
printf("%s opened succesfully!", szStr);
SetupComm(hCom, 1024, 1024);
COMMTIMEOUTS TimeOuts;
TimeOuts.ReadIntervalTimeout = MAXDWORD; /* Maximum time between read chars. */
TimeOuts.ReadTotalTimeoutMultiplier = 0; /* Multiplier of characters. */
TimeOuts.ReadTotalTimeoutConstant = 0; /* Constant in milliseconds. */
TimeOuts.WriteTotalTimeoutMultiplier = 500; /* Multiplier of characters. */
TimeOuts.WriteTotalTimeoutConstant = 100; /* Constant in milliseconds. */
SetCommTimeouts(hCom, &TimeOuts);
DCB dcb = {0};
dcb.BaudRate = 115200;
dcb.ByteSize = 8;
dcb.Parity = NOPARITY;
dcb.StopBits = ONESTOPBIT;
SetCommState(hCom, &dcb);
PurgeComm(hCom, PURGE_TXCLEAR | PURGE_RXCLEAR | PURGE_TXABORT | PURGE_RXABORT);
return hCom;
}
int main(int argc, char **argv)
{
DWORD recvsize = 0;
char recvbuf[BUFF_SIZE];
char buf[BUFF_SIZE];
DWORD dwError;
DWORD threadId;
bool poweroff = false;
bool reboot = false;
unsigned int retry_times;
if (argc > 2)
return -1;
if ((argc == 2) && (sizeof(argv[1]) + 5 > BUFF_SIZE)) {
printf("VM name (%s) is too long\n", argv[1]);
return -1;
}
hCom2 = initCom("COM2");
if (hCom2 == INVALID_HANDLE_VALUE)
return -1;
memset(buf, 0, sizeof(buf));
memset(resend_buf, 0, sizeof(resend_buf));
CreateThread(NULL, 0, open_socket_server, NULL, 0, &threadId);
if (ClearCommError(hCom2, &dwError, NULL)) {
PurgeComm(hCom2, PURGE_TXABORT | PURGE_TXCLEAR);
}
snprintf(buf, sizeof(buf), SYNC_FMT, (argc == 1) ? WIN_VM_NAME : argv[1]);
start_uart_resend(buf, MIN_RESEND_TIME);
send_message_by_uart(hCom2, buf, strlen(buf));
/**
* The lifecycle manager in Service VM checks sync message every 5 seconds
* during listening phase, delay 5 seconds to wait Service VM to receive the
* sync message, then start to read ack message from Service VM.
*/
Sleep(5U * SECOND_TO_MS);
do {
do {
retry_times = RETRY_RECV_TIMES;
memset(recvbuf, 0, sizeof(recvbuf));
do {
ReadFile(hCom2, recvbuf, sizeof(recvbuf), &recvsize, NULL);
Sleep(READ_INTERVAL);
retry_times--;
} while ((recvsize < MSG_SIZE) && (retry_times > 0));
if (recvsize < MSG_SIZE) {
if (resend_time > 1U) {
Sleep(6U * SECOND_TO_MS);
printf("Resend command (%s) service VM\n", resend_buf);
send_message_by_uart(hCom2, resend_buf, strlen(resend_buf));
resend_time--;
} else if (resend_time == 1U) {
printf("Failed to resend command (%s)\n", resend_buf);
break;
} else {
/* No action if resend_time is 0 */
}
}
} while (recvsize < MSG_SIZE);
if (resend_time == 1U)
break;
if (recvbuf[recvsize - 1] == '\n')
recvbuf[recvsize - 1] = '\0';
if (strncmp(recvbuf, ACK_SYNC, sizeof(ACK_SYNC)) == 0)
{
stop_uart_resend();
printf("Received acked sync message from service VM\n");
} else if (strncmp(recvbuf, ACK_REQ_SYS_SHUTDOWN, sizeof(ACK_REQ_SYS_SHUTDOWN)) == 0) {
stop_uart_resend();
printf("Received acked system shutdown request from service VM\n");
} else if (strncmp(recvbuf, ACK_REQ_SYS_REBOOT, sizeof(ACK_REQ_SYS_REBOOT)) == 0) {
stop_uart_resend();
printf("Received acked system reboot request from service VM\n");
} else if (strncmp(recvbuf, POWEROFF_CMD, sizeof(POWEROFF_CMD)) == 0) {
printf("Received system shutdown message from service VM\n");
send_message_by_uart(hCom2, ACK_POWEROFF, sizeof(ACK_POWEROFF));
Sleep(2 * READ_INTERVAL);
printf("Windows VM will shutdown.\n");
poweroff = true;
break;
} else if (strncmp(recvbuf, USER_VM_SHUTDOWN, sizeof(USER_VM_SHUTDOWN)) == 0) {
printf("Received guest shutdown message from service VM\n");
send_message_by_uart(hCom2, ACK_USER_VM_SHUTDOWN, sizeof(ACK_USER_VM_SHUTDOWN));
Sleep(2 * READ_INTERVAL);
printf("Windows VM will shutdown.\n");
poweroff = true;
break;
} else if (strncmp(recvbuf, USER_VM_REBOOT, sizeof(USER_VM_REBOOT)) == 0) {
printf("Received guest reboot message from service VM\n");
send_message_by_uart(hCom2, ACK_USER_VM_REBOOT, sizeof(ACK_USER_VM_REBOOT));
Sleep(2 * READ_INTERVAL);
printf("Windows VM will reboot.\n");
reboot = true;
break;
} else {
printf("Received invalid message (%s) from service VM.\n", recvbuf);
}
} while (1);
CloseHandle(hCom2);
if (poweroff)
system("shutdown -s -t 0");
if (reboot)
system("shutdown -r -t 0");
return 0;
}