42 #include <netinet/in.h> 43 #include <sys/socket.h> 44 #include <sys/types.h> 54 #include "debug/VIO9P.hh" 55 #include "debug/VIO9PData.hh" 56 #include "params/VirtIO9PBase.hh" 57 #include "params/VirtIO9PDiod.hh" 58 #include "params/VirtIO9PProxy.hh" 59 #include "params/VirtIO9PSocket.hh" 72 #define P9MSG(type, name) \ 73 { (type), P9MsgInfo((type), "T" # name ) }, \ 74 { (type + 1), P9MsgInfo((type + 1), "R" # name ) } 88 P9MSG(32, XATTRCREATE),
117 sizeof(
Config) + params->tag.size(),
124 memcpy(
config->tag, params->tag.c_str(), params->tag.size());
143 DPRINTF(VIO9P,
"Got input data descriptor (len: %i)\n", desc->
size());
144 DPRINTF(VIO9P,
"\tPending transactions: %i\n", parent.pendingTransactions.size());
147 desc->
chainRead(0, (uint8_t *)&header,
sizeof(header));
148 header =
p9toh(header);
154 parent.pendingTransactions[header.tag] = desc;
157 parent.dumpMsg(header,
data,
sizeof(
data));
160 parent.recvTMsg(header,
data,
sizeof(
data));
166 DPRINTF(VIO9P,
"Sending RMsg\n");
169 assert(header.
len >=
sizeof(header));
177 out_desc = out_desc->
next();
179 panic(
"sendRMsg: Framing error, no output descriptor.\n");
184 out_desc->
chainWrite(0, (uint8_t *)&header_out,
sizeof(header_out));
185 out_desc->
chainWrite(
sizeof(header_out), data, size);
198 const P9MsgInfoMap::const_iterator it_msg(
p9_msg_info.find(header.
type));
201 DPRINTF(VIO9P,
"P9Msg[len = %i, type = %s (%i), tag = %i]\n",
204 DPRINTF(VIO9P,
"P9Msg[len = %i, type = Unknown (%i), tag = %i]\n",
207 DDUMP(VIO9PData, data, size);
226 warn(
"Serializing VirtIO9Base device after device has been used. It is " 227 "likely that state will be lost, and that the device will cease " 241 warn(
"Unserializing VirtIO9Base device after device has been used. It is " 242 "likely that state has been lost, and that the device will cease " 251 const uint8_t *
data,
size_t size)
254 assert(header.
len ==
sizeof(header) + size);
257 uint8_t out[header.
len];
259 memcpy(out, (uint8_t *)&header_out,
sizeof(header_out));
260 memcpy(out +
sizeof(header_out), data, size);
261 writeAll(out,
sizeof(header_out) + size);
268 readAll((uint8_t *)&header,
sizeof(header));
269 header =
p9toh(header);
271 const ssize_t payload_len(header.len -
sizeof(header));
273 panic(
"Payload length is negative!\n");
274 uint8_t
data[payload_len];
277 sendRMsg(header, data, payload_len);
286 while ((ret =
read(data, len)) == -EAGAIN)
289 panic(
"readAll: Read failed: %i\n", -ret);
301 while ((ret =
write(data, len)) == -EAGAIN)
304 panic(
"writeAll: write failed: %i\n", -ret);
315 fd_to_diod(-1), fd_from_diod(-1), diod_pid(-1)
341 const int DIOD_RFD = 3;
342 const int DIOD_WFD = 4;
344 const char *diod(p->diod.c_str());
346 DPRINTF(VIO9P,
"Using diod at %s \n", p->diod.c_str());
348 if (pipe(pipe_rfd) == -1 || pipe(pipe_wfd) == -1)
349 panic(
"Failed to create DIOD pipes: %i\n", errno);
355 int socket_id = socket(AF_UNIX, SOCK_STREAM, 0);
356 if (socket_id == -1) {
357 panic(
"Socket creation failed %i \n", errno);
360 struct sockaddr_un socket_address;
361 memset(&socket_address, 0,
sizeof(
struct sockaddr_un));
362 socket_address.sun_family = AF_UNIX;
364 const std::string socket_path =
simout.
resolve(p->socketPath);
366 " output directory an absolute path, else diod will fail!\n");
369 fatal_if(
sizeof(socket_address.sun_path) <= socket_path.length(),
370 "Incorrect length of socket path");
371 strncpy(socket_address.sun_path, socket_path.c_str(),
372 sizeof(socket_address.sun_path) - 1);
373 if (bind(socket_id, (
struct sockaddr*) &socket_address,
374 sizeof(
struct sockaddr_un)) == -1){
375 perror(
"Socket binding");
376 panic(
"Socket binding to %i failed - most likely the output dir" \
377 " and hence unused socket already exists \n", socket_id);
382 panic(
"Fork failed: %i\n", errno);
386 if (dup2(pipe_rfd[0], DIOD_RFD) == -1 ||
387 dup2(pipe_wfd[1], DIOD_WFD) == -1) {
389 panic(
"Failed to setup read/write pipes: %i\n",
395 "-d",
DTRACE(VIO9P) ?
"1" :
"0",
399 "-e", p->root.c_str(),
402 "-l", socket_path.c_str(),
404 perror(
"Starting DIOD");
405 panic(
"Failed to execute diod to %s: %i\n",socket_path, errno);
409 inform(
"Started diod with PID %u, you might need to manually kill " \
410 " diod if gem5 crashes \n",
diod_pid);
422 return ret < 0 ? -errno : ret;
430 return ret < 0 ? -errno : ret;
436 parent.serverDataReady();
447 perror(
"Killing diod process");
448 warn(
"Failed to kill diod using SIGTERM");
453 for (
unsigned i = 0;
i < 5;
i++) {
454 int wait_return = waitpid(
diod_pid, NULL, WNOHANG);
458 }
else if (wait_return == 0) {
464 warn(
"Failed in waitpid");
469 inform(
"Trying to kill diod with SIGKILL as SIGTERM failed \n");
471 perror(
"Killing diod process");
472 warn(
"Failed to kill diod using SIGKILL");
480 VirtIO9PDiodParams::create()
511 struct addrinfo hints, *result;
512 memset(&hints, 0,
sizeof(hints));
513 hints.ai_family = AF_UNSPEC;
514 hints.ai_socktype = SOCK_STREAM;
516 hints.ai_protocol = 0;
518 if ((ret = getaddrinfo(p.server.c_str(), p.port.c_str(),
519 &hints, &result)) != 0)
520 panic(
"getaddrinfo: %s\n", gai_strerror(ret));
522 DPRINTF(VIO9P,
"Connecting to 9p server '%s'.\n", p.server);
523 for (
struct addrinfo *rp = result; rp; rp = rp->ai_next) {
524 fdSocket = socket(rp->ai_family, rp->ai_socktype,
528 }
else if (connect(
fdSocket, rp->ai_addr, rp->ai_addrlen) != -1) {
536 freeaddrinfo(result);
539 panic(
"Failed to connect to 9p server (%s:%s)", p.server, p.port);
545 panic(
"9P Socket disconnected!\n");
554 ret = ::recv(
fdSocket, (
void *)data, len, 0);
558 return ret < 0 ? -errno : ret;
565 int ret(::send(
fdSocket, (
const void *)data, len, 0));
566 return ret < 0 ? -errno : ret;
572 parent.serverDataReady();
577 VirtIO9PSocketParams::create()
#define panic(...)
This implements a cprintf based panic() function.
void startup()
startup() is the final initialization call before simulation.
ssize_t read(uint8_t *data, size_t len)
Read data from the server behind the proxy.
This class implements a VirtIO transport layer for the 9p network file system.
Base class for all VirtIO-based devices.
std::string resolve(const std::string &name) const
Returns relative file names prepended with name of this directory.
VirtDescriptor * next() const
Get the pointer to the next descriptor in a chain.
ssize_t read(uint8_t *data, size_t len)
Read data from the server behind the proxy.
VirtIO 9p proxy that communicates with a 9p server over tcp sockets.
void socketDisconnect()
9p server disconnect notification
std::unique_ptr< DiodDataEvent > dataEvent
virtual ssize_t read(uint8_t *data, size_t len)=0
Read data from the server behind the proxy.
void terminateDiod()
Kill the diod child process at the end of the simulation.
void serialize(CheckpointOut &cp) const override
Serialize an object.
#define DDUMP(x, data, count)
std::map< P9Tag, VirtDescriptor * > pendingTransactions
Map between 9p transaction tags and descriptors where they appeared.
T p9toh(T v)
Convert p9 byte order (LE) to host byte order.
int fd_to_diod
fd for data pipe going to diod (write end)
void readConfig(PacketPtr pkt, Addr cfgOffset)
Read from the configuration space of a device.
void sendRMsg(const P9MsgHeader &header, const uint8_t *data, size_t size)
Send a 9p RPC message reply.
std::map< P9MsgType, P9MsgInfo > P9MsgInfoMap
virtual ssize_t write(const uint8_t *data, size_t len)=0
Write data to the server behind the proxy.
#define P9MSG(type, name)
size_t size() const
Retrieve the size of this descriptor.
int fd_from_diod
fd for data pipe coming from diod (read end)
void serialize(CheckpointOut &cp) const override
Serialize an object.
void schedule(PollEvent *event)
static const P9MsgInfoMap p9_msg_info
void produceDescriptor(VirtDescriptor *desc, uint32_t len)
Send a descriptor chain to the guest.
void chainRead(size_t offset, uint8_t *dst, size_t size) const
Read the contents of a descriptor chain.
void writeAll(const uint8_t *data, size_t len)
Convenience function that writes exactly len bytes.
std::unique_ptr< SocketDataEvent > dataEvent
#define UNSERIALIZE_SCALAR(scalar)
ByteOrder byteOrder
The byte order of the queues, descriptors, etc.
virtual ~VirtIO9PSocket()
void startup()
startup() is the final initialization call before simulation.
T htog(T value, ByteOrder guest_byte_order)
VirtIO9PDiod(Params *params)
void serverDataReady()
Notification of pending data from server.
VirtIO9PSocket(Params *params)
void kick()
Inform the guest of available buffers.
void chainWrite(size_t offset, const uint8_t *src, size_t size)
Write to a descriptor chain.
bool deviceUsed
Bool to track if the device has been used or not.
void onNotifyDescriptor(VirtDescriptor *desc)
Notify queue of pending incoming descriptor.
VirtIO descriptor (chain) wrapper.
ssize_t write(const uint8_t *data, size_t len)
Write data to the server behind the proxy.
void registerExitCallback(Callback *callback)
Register an exit callback.
#define fatal_if(cond,...)
Conditional fatal macro that checks the supplied condition and only causes a fatal error if the condi...
P9MsgInfo(P9MsgType _type, std::string _name)
void recvTMsg(const P9MsgHeader &header, const uint8_t *data, size_t size) override
Handle incoming 9p RPC message.
const Params * params() const
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
A Packet is used to encapsulate a transfer between two objects in the memory system (e...
VirtIO9PBaseParams Params
#define SERIALIZE_SCALAR(scalar)
void startDiod()
Start diod and setup the communication pipes.
void registerQueue(VirtQueue &queue)
Register a new VirtQueue with the device model.
void readAll(uint8_t *data, size_t len)
Convenience function that reads exactly len bytes.
T htop9(T v)
Convert host byte order to p9 byte order (LE)
std::ostream CheckpointOut
bool isOutgoing() const
Check if this is a write-only descriptor (outgoing data).
VirtIO 9p configuration structure.
static bool isAbsolute(const std::string &name)
Test if a path is absolute.
VirtIO9PBase(Params *params)
int diod_pid
PID of diod process.
void unserialize(CheckpointIn &cp) override
Unserialize an object.
void connectSocket()
Try to resolve the server name and connect to the 9p server.
Helper template class to turn a simple class member function into a callback.
void dumpMsg(const P9MsgHeader &header, const uint8_t *data, size_t size)
Dump a 9p RPC message on the debug output.
VirtIO 9p proxy that communicates with the diod 9p server using pipes.
void unserialize(CheckpointIn &cp) override
Unserialize an object.
std::unique_ptr< Config > config
Currently active configuration (host byte order)
void readConfigBlob(PacketPtr pkt, Addr cfgOffset, const uint8_t *cfg)
Read configuration data from a device structure.
VirtIO9PProxy(Params *params)
int fdSocket
Socket connected to the 9p server.
const size_t configSize
Size of the device's configuration space.
ByteOrder byteOrder(const ThreadContext *tc)
VirtIO 9p proxy base class.
ssize_t write(const uint8_t *data, size_t len)
Write data to the server behind the proxy.