blob: 75096a5c8ece2e6e8887b24776db23593a6c0220 [file] [log] [blame]
Andrew Scull18834872018-10-12 11:48:09 +01001/*
2 * Copyright 2018 Google LLC
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * https://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Andrew Scullfbc938a2018-08-20 14:09:28 +010017#pragma once
Wedson Almeida Filho987c0ff2018-06-20 16:34:38 +010018
Andrew Scull18c78fc2018-08-20 12:57:41 +010019#include "hf/cpu.h"
Wedson Almeida Filhoea62e2e2019-01-09 19:14:59 +000020#include "hf/list.h"
Andrew Scull18c78fc2018-08-20 12:57:41 +010021#include "hf/mm.h"
Wedson Almeida Filho22d5eaa2018-12-16 00:38:49 +000022#include "hf/mpool.h"
Wedson Almeida Filho987c0ff2018-06-20 16:34:38 +010023
Andrew Scullaa039b32018-10-04 15:02:26 +010024enum mailbox_state {
Andrew Walbranc3910f72018-11-27 14:24:36 +000025 /** There is no message in the mailbox. */
Andrew Scullaa039b32018-10-04 15:02:26 +010026 mailbox_state_empty,
27
Andrew Walbranc3910f72018-11-27 14:24:36 +000028 /** There is a message in the mailbox that is waiting for a reader. */
Andrew Scullaa039b32018-10-04 15:02:26 +010029 mailbox_state_received,
30
Andrew Walbranc3910f72018-11-27 14:24:36 +000031 /** There is a message in the mailbox that has been read. */
Andrew Scullaa039b32018-10-04 15:02:26 +010032 mailbox_state_read,
Wedson Almeida Filho2f94ec12018-07-26 16:00:48 +010033};
34
Wedson Almeida Filhoea62e2e2019-01-09 19:14:59 +000035struct wait_entry {
36 /** The VM that is waiting for a mailbox to become writable. */
37 struct vm *waiting_vm;
38
39 /**
40 * Links used to add entry to a VM's waiter_list. This is protected by
41 * the notifying VM's lock.
42 */
43 struct list_entry wait_links;
44
45 /**
46 * Links used to add entry to a VM's ready_list. This is protected by
47 * the waiting VM's lock.
48 */
49 struct list_entry ready_links;
50};
51
Andrew Scullaa039b32018-10-04 15:02:26 +010052struct mailbox {
53 enum mailbox_state state;
54 uint32_t recv_from_id;
Wedson Almeida Filho2f94ec12018-07-26 16:00:48 +010055 int16_t recv_bytes;
56 void *recv;
57 const void *send;
58 struct vcpu *recv_waiter;
Wedson Almeida Filhoea62e2e2019-01-09 19:14:59 +000059
60 /**
61 * List of wait_entry structs representing VMs that want to be notified
62 * when the mailbox becomes writable. Once the mailbox does become
63 * writable, the entry is removed from this list and added to the
64 * waiting VM's ready_list.
65 */
66 struct list_entry waiter_list;
67
68 /**
69 * List of wait_entry structs representing VMs whose mailboxes became
70 * writable since the owner of the mailbox registers for notification.
71 */
72 struct list_entry ready_list;
Wedson Almeida Filho2f94ec12018-07-26 16:00:48 +010073};
74
Wedson Almeida Filho987c0ff2018-06-20 16:34:38 +010075struct vm {
Andrew Scull8c3a63a2018-09-20 13:38:34 +010076 uint32_t id;
Andrew Walbran0d7a0682018-12-06 16:48:47 +000077 /** See api.c for the partial ordering on locks. */
Wedson Almeida Filho2f94ec12018-07-26 16:00:48 +010078 struct spinlock lock;
Wedson Almeida Filho87009642018-07-02 10:20:07 +010079 uint32_t vcpu_count;
Wedson Almeida Filho84a30a02018-07-23 20:05:05 +010080 struct vcpu vcpus[MAX_CPUS];
Andrew Scull89a75242018-08-06 17:04:55 +010081 struct mm_ptable ptable;
Andrew Scullaa039b32018-10-04 15:02:26 +010082 struct mailbox mailbox;
Wedson Almeida Filhoea62e2e2019-01-09 19:14:59 +000083
84 /** Wait entries to be used when waiting on other VM mailboxes. */
85 struct wait_entry wentry[MAX_VMS];
86};
87
88/** Encapsulates a VM whose lock is held. */
89struct vm_locked {
90 struct vm *vm;
Wedson Almeida Filho987c0ff2018-06-20 16:34:38 +010091};
92
Wedson Almeida Filho22d5eaa2018-12-16 00:38:49 +000093bool vm_init(uint32_t vcpu_count, struct mpool *ppool, struct vm **new_vm);
Andrew Scull19503262018-09-20 14:48:39 +010094uint32_t vm_get_count(void);
95struct vm *vm_get(uint32_t id);
Andrew Scull37402872018-10-24 14:23:06 +010096void vm_start_vcpu(struct vm *vm, size_t index, ipaddr_t entry, uintreg_t arg);
Wedson Almeida Filhoea62e2e2019-01-09 19:14:59 +000097void vm_lock(struct vm *vm, struct vm_locked *locked);
98void vm_unlock(struct vm_locked *locked);