Go to the documentation of this file.
42 #include <netinet/in.h>
43 #include <sys/socket.h>
44 #include <sys/types.h>
55 #include "debug/VIO9P.hh"
56 #include "debug/VIO9PData.hh"
57 #include "params/VirtIO9PBase.hh"
58 #include "params/VirtIO9PDiod.hh"
59 #include "params/VirtIO9PProxy.hh"
60 #include "params/VirtIO9PSocket.hh"
78 #define P9MSG(type, name) \
79 { (type), P9MsgInfo((type), "T" # name ) }, \
80 { (type + 1), P9MsgInfo((type + 1), "R" # name ) }
94 P9MSG(32, XATTRCREATE),
123 sizeof(
Config) + params.tag.size(),
149 DPRINTF(VIO9P,
"Got input data descriptor (len: %i)\n", desc->
size());
172 DPRINTF(VIO9P,
"Sending RMsg\n");
183 out_desc = out_desc->
next();
185 panic(
"sendRMsg: Framing error, no output descriptor.\n");
190 out_desc->
chainWrite(0, (uint8_t *)&header_out,
sizeof(header_out));
207 DPRINTF(VIO9P,
"P9Msg[len = %i, type = %s (%i), tag = %i]\n",
210 DPRINTF(VIO9P,
"P9Msg[len = %i, type = Unknown (%i), tag = %i]\n",
232 warn(
"Serializing VirtIO9Base device after device has been used. It is "
233 "likely that state will be lost, and that the device will cease "
247 warn(
"Unserializing VirtIO9Base device after device has been used. It is "
248 "likely that state has been lost, and that the device will cease "
257 const uint8_t *
data,
size_t size)
265 memcpy(out, (uint8_t *)&header_out,
sizeof(header_out));
266 memcpy(out +
sizeof(header_out),
data, size);
267 writeAll(out,
sizeof(header_out) + size);
279 panic(
"Payload length is negative!\n");
280 uint8_t
data[payload_len];
295 panic(
"readAll: Read failed: %i\n", -ret);
310 panic(
"writeAll: write failed: %i\n", -ret);
321 fd_to_diod(-1), fd_from_diod(-1), diod_pid(-1)
346 DPRINTF(VIO9P,
"Using diod at %s.\n",
p.diod);
348 panic_if(pipe(pipe_rfd) == -1,
"Failed to create DIOD read pipe: %s",
350 panic_if(pipe(pipe_wfd) == -1,
"Failed to create DIOD write pipe: %s",
357 int socket_id = socket(AF_UNIX, SOCK_STREAM, 0);
358 panic_if(socket_id == -1,
"Socket creation failed %i", errno);
361 struct sockaddr_un socket_address;
362 memset(&socket_address, 0,
sizeof(socket_address));
363 socket_address.sun_family = AF_UNIX;
367 " output directory an absolute path, else diod will fail!");
370 fatal_if(
sizeof(socket_address.sun_path) <= socket_path.length(),
371 "Incorrect length of socket path");
372 strncpy(socket_address.sun_path, socket_path.c_str(),
373 sizeof(socket_address.sun_path) - 1);
374 panic_if(bind(socket_id, (
struct sockaddr*)&socket_address,
375 sizeof(socket_address)) == -1,
376 "Socket binding to %i failed - most likely the output dir "
377 "and hence unused socket already exists.", socket_id);
392 execlp(
p.diod.c_str(),
p.diod.c_str(),
393 "-d", debug::VIO9P ?
"1" :
"0",
395 "-r", diod_rfd_s.c_str(),
396 "-w", diod_wfd_s.c_str(),
397 "-e",
p.root.c_str(),
400 "-l", socket_path.c_str(),
402 panic(
"Failed to execute diod to %s: %s", socket_path,
407 inform(
"Started diod with PID %u, you might need to manually kill "
417 return ret < 0 ? -errno : ret;
425 return ret < 0 ? -errno : ret;
442 perror(
"Killing diod process");
443 warn(
"Failed to kill diod using SIGTERM");
448 for (
unsigned i = 0;
i < 5;
i++) {
449 int wait_return = waitpid(
diod_pid, NULL, WNOHANG);
453 }
else if (wait_return == 0) {
459 warn(
"Failed in waitpid");
464 inform(
"Trying to kill diod with SIGKILL as SIGTERM failed \n");
466 perror(
"Killing diod process");
467 warn(
"Failed to kill diod using SIGKILL");
498 struct addrinfo hints, *result;
499 memset(&hints, 0,
sizeof(hints));
500 hints.ai_family = AF_UNSPEC;
501 hints.ai_socktype = SOCK_STREAM;
503 hints.ai_protocol = 0;
505 if ((ret = getaddrinfo(
p.server.c_str(),
p.port.c_str(),
506 &hints, &result)) != 0)
507 panic(
"getaddrinfo: %s\n", gai_strerror(ret));
509 DPRINTF(VIO9P,
"Connecting to 9p server '%s'.\n",
p.server);
510 for (
struct addrinfo *rp = result; rp; rp = rp->ai_next) {
511 fdSocket = socket(rp->ai_family, rp->ai_socktype,
515 }
else if (connect(
fdSocket, rp->ai_addr, rp->ai_addrlen) != -1) {
523 freeaddrinfo(result);
526 panic(
"Failed to connect to 9p server (%s:%s)",
p.server,
p.port);
532 panic(
"9P Socket disconnected!\n");
545 return ret < 0 ? -errno : ret;
553 return ret < 0 ? -errno : ret;
This class implements a VirtIO transport layer for the 9p network file system.
std::unique_ptr< DiodDataEvent > dataEvent
int fdSocket
Socket connected to the 9p server.
ssize_t read(uint8_t *data, size_t len)
Read data from the server behind the proxy.
VirtIO9PBaseParams Params
std::map< P9Tag, VirtDescriptor * > pendingTransactions
Map between 9p transaction tags and descriptors where they appeared.
static bool isAbsolute(const std::string &name)
Test if a path is absolute.
#define UNSERIALIZE_SCALAR(scalar)
void unserialize(CheckpointIn &cp) override
Unserialize an object.
void writeAll(const uint8_t *data, size_t len)
Convenience function that writes exactly len bytes.
void unserialize(CheckpointIn &cp) override
Unserialize an object.
void serialize(CheckpointOut &cp) const override
Serialize an object.
const std::string to_string(sc_enc enc)
const size_t configSize
Size of the device's configuration space.
bool deviceUsed
Bool to track if the device has been used or not.
ByteOrder byteOrder(const ThreadContext *tc)
void chainWrite(size_t offset, const uint8_t *src, size_t size)
Write to a descriptor chain.
#define P9MSG(type, name)
void readConfigBlob(PacketPtr pkt, Addr cfgOffset, const uint8_t *cfg)
Read configuration data from a device structure.
VirtDescriptor * next() const
Get the pointer to the next descriptor in a chain.
void socketDisconnect()
9p server disconnect notification
virtual void recvTMsg(const P9MsgHeader &header, const uint8_t *data, size_t size)=0
Handle incoming 9p RPC message.
bool isOutgoing() const
Check if this is a write-only descriptor (outgoing data).
VirtIO9PSocket(const Params ¶ms)
void serialize(CheckpointOut &cp) const override
Serialize an object.
void onNotifyDescriptor(VirtDescriptor *desc)
Notify queue of pending incoming descriptor.
const Params & params() const
int fd_to_diod
fd for data pipe going to diod (write end)
void registerQueue(VirtQueue &queue)
Register a new VirtQueue with the device model.
A Packet is used to encapsulate a transfer between two objects in the memory system (e....
std::map< P9MsgType, P9MsgInfo > P9MsgInfoMap
void kick()
Inform the guest of available buffers.
void startup()
startup() is the final initialization call before simulation.
void readAll(uint8_t *data, size_t len)
Convenience function that reads exactly len bytes.
size_t size() const
Retrieve the size of this descriptor.
std::string resolve(const std::string &name) const
Returns relative file names prepended with name of this directory.
std::unique_ptr< SocketDataEvent > dataEvent
T htog(T value, ByteOrder guest_byte_order)
void sendRMsg(const P9MsgHeader &header, const uint8_t *data, size_t size)
Send a 9p RPC message reply.
void schedule(PollEvent *event)
Base class for all VirtIO-based devices.
VirtIO9PBase(const Params ¶ms)
T p9toh(T v)
Convert p9 byte order (LE) to host byte order.
void startup()
startup() is the final initialization call before simulation.
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
#define SERIALIZE_SCALAR(scalar)
#define DDUMP(x, data, count)
DPRINTF is a debugging trace facility that allows one to selectively enable tracing statements.
int fd_from_diod
fd for data pipe coming from diod (read end)
void terminateDiod()
Kill the diod child process at the end of the simulation.
void readConfig(PacketPtr pkt, Addr cfgOffset)
Read from the configuration space of a device.
T htop9(T v)
Convert host byte order to p9 byte order (LE)
static const P9MsgInfoMap p9_msg_info
#define panic_if(cond,...)
Conditional panic macro that checks the supplied condition and only panics if the condition is true a...
void recvTMsg(const P9MsgHeader &header, const uint8_t *data, size_t size) override
Handle incoming 9p RPC message.
VirtIO 9p configuration structure.
ssize_t read(uint8_t *data, size_t len)
Read data from the server behind the proxy.
VirtIO9PDiod(const Params ¶ms)
VirtIO descriptor (chain) wrapper.
std::unique_ptr< Config > config
Currently active configuration (host byte order)
ByteOrder byteOrder
The byte order of the queues, descriptors, etc.
void produceDescriptor(VirtDescriptor *desc, uint32_t len)
Send a descriptor chain to the guest.
ssize_t write(const uint8_t *data, size_t len)
Write data to the server behind the proxy.
void chainRead(size_t offset, uint8_t *dst, size_t size) const
Read the contents of a descriptor chain.
void dumpMsg(const P9MsgHeader &header, const uint8_t *data, size_t size)
Dump a 9p RPC message on the debug output.
void connectSocket()
Try to resolve the server name and connect to the 9p server.
std::ostream CheckpointOut
VirtIO9PProxy(const Params ¶ms)
void startDiod()
Start diod and setup the communication pipes.
virtual ssize_t write(const uint8_t *data, size_t len)=0
Write data to the server behind the proxy.
ssize_t write(const uint8_t *data, size_t len)
Write data to the server behind the proxy.
VirtIO 9p proxy base class.
P9MsgInfo(P9MsgType _type, std::string _name)
void registerExitCallback(const std::function< void()> &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...
Reference material can be found at the JEDEC website: UFS standard http://www.jedec....
void serverDataReady()
Notification of pending data from server.
int diod_pid
PID of diod process.
virtual ssize_t read(uint8_t *data, size_t len)=0
Read data from the server behind the proxy.
#define panic(...)
This implements a cprintf based panic() function.
virtual ~VirtIO9PSocket()
Generated on Wed Jul 28 2021 12:10:27 for gem5 by doxygen 1.8.17