gem5  v20.0.0.3
fs9p.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2014-2017 ARM Limited
3  * All rights reserved
4  *
5  * The license below extends only to copyright in the software and shall
6  * not be construed as granting a license to any other intellectual
7  * property including but not limited to intellectual property relating
8  * to a hardware implementation of the functionality of the software
9  * licensed hereunder. You may use the software subject to the license
10  * terms below provided that you ensure that this notice is replicated
11  * unmodified and in its entirety in all distributions of the software,
12  * modified or unmodified, in source code or in binary form.
13  *
14  * Redistribution and use in source and binary forms, with or without
15  * modification, are permitted provided that the following conditions are
16  * met: redistributions of source code must retain the above copyright
17  * notice, this list of conditions and the following disclaimer;
18  * redistributions in binary form must reproduce the above copyright
19  * notice, this list of conditions and the following disclaimer in the
20  * documentation and/or other materials provided with the distribution;
21  * neither the name of the copyright holders nor the names of its
22  * contributors may be used to endorse or promote products derived from
23  * this software without specific prior written permission.
24  *
25  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
26  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
27  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
28  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
29  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
30  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
31  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
32  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
33  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
34  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
35  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
36  */
37 
38 #include "dev/virtio/fs9p.hh"
39 
40 #include <fcntl.h>
41 #include <netdb.h>
42 #include <netinet/in.h>
43 #include <sys/socket.h>
44 #include <sys/types.h>
45 #include <sys/un.h>
46 #include <sys/wait.h>
47 #include <unistd.h>
48 
49 #include <csignal>
50 #include <fstream>
51 
52 #include "base/callback.hh"
53 #include "base/output.hh"
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"
60 #include "sim/system.hh"
61 
62 struct P9MsgInfo {
63  P9MsgInfo(P9MsgType _type, std::string _name)
64  : type(_type), name(_name) {}
65 
67  std::string name;
68 };
69 
70 typedef std::map<P9MsgType, P9MsgInfo> P9MsgInfoMap;
71 
72 #define P9MSG(type, name) \
73  { (type), P9MsgInfo((type), "T" # name ) }, \
74  { (type + 1), P9MsgInfo((type + 1), "R" # name ) }
75 
76 static const P9MsgInfoMap p9_msg_info {
77  P9MSG(6, LERROR),
78  P9MSG(8, STATFS),
79  P9MSG(12, LOPEN),
80  P9MSG(14, LCREATE),
81  P9MSG(16, SYMLINK),
82  P9MSG(18, MKNOD),
83  P9MSG(20, RENAME),
84  P9MSG(22, READLINK),
85  P9MSG(24, GETATTR),
86  P9MSG(26, SETATTR),
87  P9MSG(30, XATTRWALK),
88  P9MSG(32, XATTRCREATE),
89  P9MSG(40, READDIR),
90  P9MSG(50, FSYNC),
91  P9MSG(52, LOCK),
92  P9MSG(54, GETLOCK),
93  P9MSG(70, LINK),
94  P9MSG(72, MKDIR),
95  P9MSG(74, RENAMEAT),
96  P9MSG(76, UNLINKAT),
97  P9MSG(100, VERSION),
98  P9MSG(102, AUTH),
99  P9MSG(104, ATTACH),
100  P9MSG(106, ERROR),
101  P9MSG(108, FLUSH),
102  P9MSG(110, WALK),
103  P9MSG(112, OPEN),
104  P9MSG(114, CREATE),
105  P9MSG(116, READ),
106  P9MSG(118, WRITE),
107  P9MSG(120, CLUNK),
108  P9MSG(122, REMOVE),
109  P9MSG(124, STAT),
110  P9MSG(126, WSTAT),
111 };
112 
113 #undef P9MSG
114 
116  : VirtIODeviceBase(params, ID_9P,
117  sizeof(Config) + params->tag.size(),
118  F_MOUNT_TAG),
119  queue(params->system->physProxy, byteOrder, params->queueSize, *this)
120 {
121  config.reset((Config *)
122  operator new(configSize));
123  config->len = htog(params->tag.size(), byteOrder);
124  memcpy(config->tag, params->tag.c_str(), params->tag.size());
125 
127 }
128 
129 
131 {
132 }
133 
134 void
136 {
137  readConfigBlob(pkt, cfgOffset, (uint8_t *)config.get());
138 }
139 
140 void
142 {
143  DPRINTF(VIO9P, "Got input data descriptor (len: %i)\n", desc->size());
144  DPRINTF(VIO9P, "\tPending transactions: %i\n", parent.pendingTransactions.size());
145 
147  desc->chainRead(0, (uint8_t *)&header, sizeof(header));
148  header = p9toh(header);
149 
150  uint8_t data[header.len - sizeof(header)];
151  desc->chainRead(sizeof(header), data, sizeof(data));
152 
153  // Keep track of pending transactions
154  parent.pendingTransactions[header.tag] = desc;
155 
156  DPRINTF(VIO9P, "recvTMsg\n");
157  parent.dumpMsg(header, data, sizeof(data));
158 
159  // Notify device of message
160  parent.recvTMsg(header, data, sizeof(data));
161 }
162 
163 void
164 VirtIO9PBase::sendRMsg(const P9MsgHeader &header, const uint8_t *data, size_t size)
165 {
166  DPRINTF(VIO9P, "Sending RMsg\n");
167  dumpMsg(header, data, size);
168  DPRINTF(VIO9P, "\tPending transactions: %i\n", pendingTransactions.size());
169  assert(header.len >= sizeof(header));
170 
171  VirtDescriptor *main_desc(pendingTransactions[header.tag]);
172  pendingTransactions.erase(header.tag);
173 
174  // Find the first output descriptor
175  VirtDescriptor *out_desc(main_desc);
176  while (out_desc && !out_desc->isOutgoing())
177  out_desc = out_desc->next();
178  if (!out_desc)
179  panic("sendRMsg: Framing error, no output descriptor.\n");
180 
181  P9MsgHeader header_out(htop9(header));
182  header_out.len = htop9(sizeof(P9MsgHeader) + size);
183 
184  out_desc->chainWrite(0, (uint8_t *)&header_out, sizeof(header_out));
185  out_desc->chainWrite(sizeof(header_out), data, size);
186 
187  queue.produceDescriptor(main_desc, sizeof(P9MsgHeader) + size);
188  kick();
189 }
190 
191 void
192 VirtIO9PBase::dumpMsg(const P9MsgHeader &header, const uint8_t *data, size_t size)
193 {
194 #ifndef NDEBUG
195  if (!DTRACE(VIO9P))
196  return;
197 
198  const P9MsgInfoMap::const_iterator it_msg(p9_msg_info.find(header.type));
199  if (it_msg != p9_msg_info.cend()) {
200  const P9MsgInfo &info(it_msg->second);
201  DPRINTF(VIO9P, "P9Msg[len = %i, type = %s (%i), tag = %i]\n",
202  header.len, info.name, header.type, header.tag);
203  } else {
204  DPRINTF(VIO9P, "P9Msg[len = %i, type = Unknown (%i), tag = %i]\n",
205  header.len, header.type, header.tag);
206  }
207  DDUMP(VIO9PData, data, size);
208 #endif
209 }
210 
211 
213  : VirtIO9PBase(params), deviceUsed(false)
214 {
215 }
216 
218 {
219 }
220 
221 
222 void
224 {
225  if (deviceUsed) {
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 "
228  "to work!");
229  }
231 
233 }
234 
235 void
237 {
239 
240  if (deviceUsed) {
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 "
243  "to work!");
244  }
246 }
247 
248 
249 void
251  const uint8_t *data, size_t size)
252 {
253  deviceUsed = true;
254  assert(header.len == sizeof(header) + size);
255  // While technically not needed, we send the packet as one
256  // contiguous segment to make some packet dissectors happy.
257  uint8_t out[header.len];
258  P9MsgHeader header_out(htop9(header));
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);
262 }
263 
264 void
266 {
268  readAll((uint8_t *)&header, sizeof(header));
269  header = p9toh(header);
270 
271  const ssize_t payload_len(header.len - sizeof(header));
272  if (payload_len < 0)
273  panic("Payload length is negative!\n");
274  uint8_t data[payload_len];
275  readAll(data, payload_len);
276 
277  sendRMsg(header, data, payload_len);
278 }
279 
280 
281 void
282 VirtIO9PProxy::readAll(uint8_t *data, size_t len)
283 {
284  while (len) {
285  ssize_t ret;
286  while ((ret = read(data, len)) == -EAGAIN)
287  ;
288  if (ret < 0)
289  panic("readAll: Read failed: %i\n", -ret);
290 
291  len -= ret;
292  data += ret;
293  }
294 }
295 
296 void
297 VirtIO9PProxy::writeAll(const uint8_t *data, size_t len)
298 {
299  while (len) {
300  ssize_t ret;
301  while ((ret = write(data, len)) == -EAGAIN)
302  ;
303  if (ret < 0)
304  panic("writeAll: write failed: %i\n", -ret);
305 
306  len -= ret;
307  data += ret;
308  }
309 }
310 
311 
312 
314  : VirtIO9PProxy(params),
315  fd_to_diod(-1), fd_from_diod(-1), diod_pid(-1)
316 {
317  // Register an exit callback so we can kill the diod process
321 }
322 
324 {
325 }
326 
327 void
329 {
330  startDiod();
331  dataEvent.reset(new DiodDataEvent(*this, fd_from_diod, POLLIN));
333 }
334 
335 void
337 {
338  const Params *p(dynamic_cast<const Params *>(params()));
339  int pipe_rfd[2];
340  int pipe_wfd[2];
341  const int DIOD_RFD = 3;
342  const int DIOD_WFD = 4;
343 
344  const char *diod(p->diod.c_str());
345 
346  DPRINTF(VIO9P, "Using diod at %s \n", p->diod.c_str());
347 
348  if (pipe(pipe_rfd) == -1 || pipe(pipe_wfd) == -1)
349  panic("Failed to create DIOD pipes: %i\n", errno);
350 
351  fd_to_diod = pipe_rfd[1];
352  fd_from_diod = pipe_wfd[0];
353 
354  // Create Unix domain socket
355  int socket_id = socket(AF_UNIX, SOCK_STREAM, 0);
356  if (socket_id == -1) {
357  panic("Socket creation failed %i \n", errno);
358  }
359  // Bind the socket to a path which will not be read
360  struct sockaddr_un socket_address;
361  memset(&socket_address, 0, sizeof(struct sockaddr_un));
362  socket_address.sun_family = AF_UNIX;
363 
364  const std::string socket_path = simout.resolve(p->socketPath);
365  fatal_if(!OutputDirectory::isAbsolute(socket_path), "Please make the" \
366  " output directory an absolute path, else diod will fail!\n");
367 
368  // Prevent overflow in strcpy
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);
378  }
379 
380  diod_pid = fork();
381  if (diod_pid == -1) {
382  panic("Fork failed: %i\n", errno);
383  } else if (diod_pid == 0) {
384  // Create the socket which will later by used by the diod process
385  close(STDIN_FILENO);
386  if (dup2(pipe_rfd[0], DIOD_RFD) == -1 ||
387  dup2(pipe_wfd[1], DIOD_WFD) == -1) {
388 
389  panic("Failed to setup read/write pipes: %i\n",
390  errno);
391  }
392 
393  // Start diod
394  execlp(diod, diod,
395  "-d", DTRACE(VIO9P) ? "1" : "0", // show debug output
396  "-f", // start in foreground
397  "-r", "3", // setup read FD
398  "-w", "4", // setup write FD
399  "-e", p->root.c_str(), // path to export
400  "-n", // disable security
401  "-S", // squash all users
402  "-l", socket_path.c_str(), // pass the socket
403  (char *)NULL);
404  perror("Starting DIOD");
405  panic("Failed to execute diod to %s: %i\n",socket_path, errno);
406  } else {
407  close(pipe_rfd[0]);
408  close(pipe_wfd[1]);
409  inform("Started diod with PID %u, you might need to manually kill " \
410  " diod if gem5 crashes \n", diod_pid);
411  }
412 
413 #undef DIOD_RFD
414 #undef DIOD_WFD
415 }
416 
417 ssize_t
418 VirtIO9PDiod::read(uint8_t *data, size_t len)
419 {
420  assert(fd_from_diod != -1);
421  const int ret(::read(fd_from_diod, (void *)data, len));
422  return ret < 0 ? -errno : ret;
423 }
424 
425 ssize_t
426 VirtIO9PDiod::write(const uint8_t *data, size_t len)
427 {
428  assert(fd_to_diod != -1);
429  const int ret(::write(fd_to_diod, (const void *)data, len));
430  return ret < 0 ? -errno : ret;
431 }
432 
433 void
435 {
436  parent.serverDataReady();
437 }
438 
439 void
441 {
442  assert(diod_pid != -1);
443 
444  DPRINTF(VIO9P, "Trying to kill diod at pid %u \n", diod_pid);
445 
446  if (kill(diod_pid, SIGTERM) != 0) {
447  perror("Killing diod process");
448  warn("Failed to kill diod using SIGTERM");
449  return;
450  }
451 
452  // Check if kill worked
453  for (unsigned i = 0; i < 5; i++) {
454  int wait_return = waitpid(diod_pid, NULL, WNOHANG);
455  if (wait_return == diod_pid) {
456  // Managed to kill diod
457  return;
458  } else if (wait_return == 0) {
459  // Diod is not killed so sleep and try again
460  usleep(500);
461  } else {
462  // Failed in waitpid
463  perror("Waitpid");
464  warn("Failed in waitpid");
465  }
466  }
467 
468  // Try again to kill diod with sigkill
469  inform("Trying to kill diod with SIGKILL as SIGTERM failed \n");
470  if (kill(diod_pid, SIGKILL) != 0) {
471  perror("Killing diod process");
472  warn("Failed to kill diod using SIGKILL");
473  } else {
474  // Managed to kill diod
475  return;
476  }
477 
478 }
479 VirtIO9PDiod *
480 VirtIO9PDiodParams::create()
481 {
482  return new VirtIO9PDiod(this);
483 }
484 
485 
486 
487 
489  : VirtIO9PProxy(params), fdSocket(-1)
490 {
491 }
492 
494 {
495 }
496 
497 void
499 {
500  connectSocket();
501  dataEvent.reset(new SocketDataEvent(*this, fdSocket, POLLIN));
503 }
504 
505 void
507 {
508  const Params &p(dynamic_cast<const Params &>(*params()));
509 
510  int ret;
511  struct addrinfo hints, *result;
512  memset(&hints, 0, sizeof(hints));
513  hints.ai_family = AF_UNSPEC;
514  hints.ai_socktype = SOCK_STREAM;
515  hints.ai_flags = 0;
516  hints.ai_protocol = 0;
517 
518  if ((ret = getaddrinfo(p.server.c_str(), p.port.c_str(),
519  &hints, &result)) != 0)
520  panic("getaddrinfo: %s\n", gai_strerror(ret));
521 
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,
525  rp->ai_protocol);
526  if (fdSocket == -1) {
527  continue;
528  } else if (connect(fdSocket, rp->ai_addr, rp->ai_addrlen) != -1) {
529  break;
530  } else {
531  close(fdSocket);
532  fdSocket = -1;
533  }
534  }
535 
536  freeaddrinfo(result);
537 
538  if (fdSocket == -1)
539  panic("Failed to connect to 9p server (%s:%s)", p.server, p.port);
540 }
541 
542 void
544 {
545  panic("9P Socket disconnected!\n");
546 }
547 
548 ssize_t
549 VirtIO9PSocket::read(uint8_t *data, size_t len)
550 {
551  assert(fdSocket != -1);
552  int ret;
553 
554  ret = ::recv(fdSocket, (void *)data, len, 0);
555  if (ret == 0)
557 
558  return ret < 0 ? -errno : ret;
559 }
560 
561 ssize_t
562 VirtIO9PSocket::write(const uint8_t *data, size_t len)
563 {
564  assert(fdSocket != -1);
565  int ret(::send(fdSocket, (const void *)data, len, 0));
566  return ret < 0 ? -errno : ret;
567 }
568 
569 void
571 {
572  parent.serverDataReady();
573 }
574 
575 
577 VirtIO9PSocketParams::create()
578 {
579  return new VirtIO9PSocket(this);
580 }
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:163
#define DPRINTF(x,...)
Definition: trace.hh:225
void startup()
startup() is the final initialization call before simulation.
Definition: fs9p.cc:328
ssize_t read(uint8_t *data, size_t len)
Read data from the server behind the proxy.
Definition: fs9p.cc:418
This class implements a VirtIO transport layer for the 9p network file system.
Definition: fs9p.hh:107
Base class for all VirtIO-based devices.
Definition: base.hh:558
std::string resolve(const std::string &name) const
Returns relative file names prepended with name of this directory.
Definition: output.cc:201
VirtDescriptor * next() const
Get the pointer to the next descriptor in a chain.
Definition: base.cc:134
OutputDirectory simout
Definition: output.cc:61
ssize_t read(uint8_t *data, size_t len)
Read data from the server behind the proxy.
Definition: fs9p.cc:549
output header
Definition: nop.cc:36
VirtIO 9p proxy that communicates with a 9p server over tcp sockets.
Definition: fs9p.hh:342
virtual ~VirtIO9PProxy()
Definition: fs9p.cc:217
Generic callback class.
Definition: callback.hh:39
PollQueue pollQueue
Definition: pollevent.cc:55
Bitfield< 7 > i
void socketDisconnect()
9p server disconnect notification
Definition: fs9p.cc:543
std::unique_ptr< DiodDataEvent > dataEvent
Definition: fs9p.hh:330
virtual ssize_t read(uint8_t *data, size_t len)=0
Read data from the server behind the proxy.
void process(int revent)
Definition: fs9p.cc:570
void terminateDiod()
Kill the diod child process at the end of the simulation.
Definition: fs9p.cc:440
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: fs9p.cc:223
#define DDUMP(x, data, count)
Definition: trace.hh:224
uint8_t P9MsgType
Definition: fs9p.hh:48
std::map< P9Tag, VirtDescriptor * > pendingTransactions
Map between 9p transaction tags and descriptors where they appeared.
Definition: fs9p.hh:200
std::string name
Definition: fs9p.cc:67
Definition: cprintf.cc:40
T p9toh(T v)
Convert p9 byte order (LE) to host byte order.
Definition: fs9p.hh:64
int fd_to_diod
fd for data pipe going to diod (write end)
Definition: fs9p.hh:326
void readConfig(PacketPtr pkt, Addr cfgOffset)
Read from the configuration space of a device.
Definition: fs9p.cc:135
void sendRMsg(const P9MsgHeader &header, const uint8_t *data, size_t size)
Send a 9p RPC message reply.
Definition: fs9p.cc:164
std::map< P9MsgType, P9MsgInfo > P9MsgInfoMap
Definition: fs9p.cc:70
virtual ssize_t write(const uint8_t *data, size_t len)=0
Write data to the server behind the proxy.
#define P9MSG(type, name)
Definition: fs9p.cc:72
size_t size() const
Retrieve the size of this descriptor.
Definition: base.hh:202
int fd_from_diod
fd for data pipe coming from diod (read end)
Definition: fs9p.hh:328
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: base.cc:342
void schedule(PollEvent *event)
Definition: pollevent.cc:159
static const P9MsgInfoMap p9_msg_info
Definition: fs9p.cc:76
void produceDescriptor(VirtDescriptor *desc, uint32_t len)
Send a descriptor chain to the guest.
Definition: base.cc:290
void chainRead(size_t offset, uint8_t *dst, size_t size) const
Read the contents of a descriptor chain.
Definition: base.cc:168
void writeAll(const uint8_t *data, size_t len)
Convenience function that writes exactly len bytes.
Definition: fs9p.cc:297
std::unique_ptr< SocketDataEvent > dataEvent
Definition: fs9p.hh:381
#define UNSERIALIZE_SCALAR(scalar)
Definition: serialize.hh:770
ByteOrder byteOrder
The byte order of the queues, descriptors, etc.
Definition: base.hh:716
#define inform(...)
Definition: logging.hh:209
virtual ~VirtIO9PSocket()
Definition: fs9p.cc:493
P9Tag tag
Message tag.
Definition: fs9p.hh:59
#define DTRACE(x)
Definition: trace.hh:223
void startup()
startup() is the final initialization call before simulation.
Definition: fs9p.cc:498
T htog(T value, ByteOrder guest_byte_order)
Definition: byteswap.hh:155
VirtIO9PDiod(Params *params)
Definition: fs9p.cc:313
void serverDataReady()
Notification of pending data from server.
Definition: fs9p.cc:265
VirtIO9PSocket(Params *params)
Definition: fs9p.cc:488
void kick()
Inform the guest of available buffers.
Definition: base.hh:608
uint32_t len
Length including header.
Definition: fs9p.hh:55
void chainWrite(size_t offset, const uint8_t *src, size_t size)
Write to a descriptor chain.
Definition: base.cc:191
void process(int revent)
Definition: fs9p.cc:434
Bitfield< 18, 16 > len
bool deviceUsed
Bool to track if the device has been used or not.
Definition: fs9p.hh:281
void onNotifyDescriptor(VirtDescriptor *desc)
Notify queue of pending incoming descriptor.
Definition: fs9p.cc:141
VirtIO descriptor (chain) wrapper.
Definition: base.hh:106
ssize_t write(const uint8_t *data, size_t len)
Write data to the server behind the proxy.
Definition: fs9p.cc:426
void registerExitCallback(Callback *callback)
Register an exit callback.
Definition: core.cc:140
#define fatal_if(cond,...)
Conditional fatal macro that checks the supplied condition and only causes a fatal error if the condi...
Definition: logging.hh:199
FSQueue queue
Definition: fs9p.hh:161
P9MsgInfo(P9MsgType _type, std::string _name)
Definition: fs9p.cc:63
void recvTMsg(const P9MsgHeader &header, const uint8_t *data, size_t size) override
Handle incoming 9p RPC message.
Definition: fs9p.cc:250
const Params * params() const
Definition: sim_object.hh:119
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:140
A Packet is used to encapsulate a transfer between two objects in the memory system (e...
Definition: packet.hh:249
virtual ~VirtIO9PDiod()
Definition: fs9p.cc:323
Bitfield< 15 > system
Definition: misc.hh:997
VirtIO9PBaseParams Params
Definition: fs9p.hh:110
#define SERIALIZE_SCALAR(scalar)
Definition: serialize.hh:763
void startDiod()
Start diod and setup the communication pipes.
Definition: fs9p.cc:336
virtual ~VirtIO9PBase()
Definition: fs9p.cc:130
void registerQueue(VirtQueue &queue)
Register a new VirtQueue with the device model.
Definition: base.cc:478
void readAll(uint8_t *data, size_t len)
Convenience function that reads exactly len bytes.
Definition: fs9p.cc:282
T htop9(T v)
Convert host byte order to p9 byte order (LE)
Definition: fs9p.hh:68
std::ostream CheckpointOut
Definition: serialize.hh:63
bool isOutgoing() const
Check if this is a write-only descriptor (outgoing data).
Definition: base.hh:221
VirtIO 9p configuration structure.
Definition: fs9p.hh:123
P9MsgType type
Definition: fs9p.cc:66
static bool isAbsolute(const std::string &name)
Test if a path is absolute.
Definition: output.hh:274
VirtIO9PBase(Params *params)
Definition: fs9p.cc:115
int diod_pid
PID of diod process.
Definition: fs9p.hh:333
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: fs9p.cc:236
void connectSocket()
Try to resolve the server name and connect to the 9p server.
Definition: fs9p.cc:506
#define warn(...)
Definition: logging.hh:208
Helper template class to turn a simple class member function into a callback.
Definition: callback.hh:62
void dumpMsg(const P9MsgHeader &header, const uint8_t *data, size_t size)
Dump a 9p RPC message on the debug output.
Definition: fs9p.cc:192
VirtIO 9p proxy that communicates with the diod 9p server using pipes.
Definition: fs9p.hh:290
Bitfield< 0 > p
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: base.cc:352
std::unique_ptr< Config > config
Currently active configuration (host byte order)
Definition: fs9p.hh:129
P9MsgType type
Message type.
Definition: fs9p.hh:57
const char data[]
void readConfigBlob(PacketPtr pkt, Addr cfgOffset, const uint8_t *cfg)
Read configuration data from a device structure.
Definition: base.cc:421
VirtIO9PProxy(Params *params)
Definition: fs9p.cc:212
int fdSocket
Socket connected to the 9p server.
Definition: fs9p.hh:379
const size_t configSize
Size of the device&#39;s configuration space.
Definition: base.hh:843
ByteOrder byteOrder(const ThreadContext *tc)
Definition: utility.hh:437
VirtIO 9p proxy base class.
Definition: fs9p.hh:211
ssize_t write(const uint8_t *data, size_t len)
Write data to the server behind the proxy.
Definition: fs9p.cc:562

Generated on Fri Jul 3 2020 15:53:02 for gem5 by doxygen 1.8.13