spapr_llan.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563
  1. /*
  2. * QEMU PowerPC pSeries Logical Partition (aka sPAPR) hardware System Emulator
  3. *
  4. * PAPR Inter-VM Logical Lan, aka ibmveth
  5. *
  6. * Copyright (c) 2010,2011 David Gibson, IBM Corporation.
  7. *
  8. * Permission is hereby granted, free of charge, to any person obtaining a copy
  9. * of this software and associated documentation files (the "Software"), to deal
  10. * in the Software without restriction, including without limitation the rights
  11. * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  12. * copies of the Software, and to permit persons to whom the Software is
  13. * furnished to do so, subject to the following conditions:
  14. *
  15. * The above copyright notice and this permission notice shall be included in
  16. * all copies or substantial portions of the Software.
  17. *
  18. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  19. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  20. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
  21. * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  22. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  23. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  24. * THE SOFTWARE.
  25. *
  26. */
  27. #include "hw/hw.h"
  28. #include "net/net.h"
  29. #include "hw/qdev.h"
  30. #include "hw/ppc/spapr.h"
  31. #include "hw/ppc/spapr_vio.h"
  32. #include "sysemu/sysemu.h"
  33. #include <libfdt.h>
  34. #define ETH_ALEN 6
  35. #define MAX_PACKET_SIZE 65536
  36. /*#define DEBUG*/
  37. #ifdef DEBUG
  38. #define DPRINTF(fmt...) do { fprintf(stderr, fmt); } while (0)
  39. #else
  40. #define DPRINTF(fmt...)
  41. #endif
  42. /*
  43. * Virtual LAN device
  44. */
  45. typedef uint64_t vlan_bd_t;
  46. #define VLAN_BD_VALID 0x8000000000000000ULL
  47. #define VLAN_BD_TOGGLE 0x4000000000000000ULL
  48. #define VLAN_BD_NO_CSUM 0x0200000000000000ULL
  49. #define VLAN_BD_CSUM_GOOD 0x0100000000000000ULL
  50. #define VLAN_BD_LEN_MASK 0x00ffffff00000000ULL
  51. #define VLAN_BD_LEN(bd) (((bd) & VLAN_BD_LEN_MASK) >> 32)
  52. #define VLAN_BD_ADDR_MASK 0x00000000ffffffffULL
  53. #define VLAN_BD_ADDR(bd) ((bd) & VLAN_BD_ADDR_MASK)
  54. #define VLAN_VALID_BD(addr, len) (VLAN_BD_VALID | \
  55. (((len) << 32) & VLAN_BD_LEN_MASK) | \
  56. (addr & VLAN_BD_ADDR_MASK))
  57. #define VLAN_RXQC_TOGGLE 0x80
  58. #define VLAN_RXQC_VALID 0x40
  59. #define VLAN_RXQC_NO_CSUM 0x02
  60. #define VLAN_RXQC_CSUM_GOOD 0x01
  61. #define VLAN_RQ_ALIGNMENT 16
  62. #define VLAN_RXQ_BD_OFF 0
  63. #define VLAN_FILTER_BD_OFF 8
  64. #define VLAN_RX_BDS_OFF 16
  65. #define VLAN_MAX_BUFS ((SPAPR_TCE_PAGE_SIZE - VLAN_RX_BDS_OFF) / 8)
  66. #define TYPE_VIO_SPAPR_VLAN_DEVICE "spapr-vlan"
  67. #define VIO_SPAPR_VLAN_DEVICE(obj) \
  68. OBJECT_CHECK(VIOsPAPRVLANDevice, (obj), TYPE_VIO_SPAPR_VLAN_DEVICE)
  69. typedef struct VIOsPAPRVLANDevice {
  70. VIOsPAPRDevice sdev;
  71. NICConf nicconf;
  72. NICState *nic;
  73. bool isopen;
  74. target_ulong buf_list;
  75. uint32_t add_buf_ptr, use_buf_ptr, rx_bufs;
  76. target_ulong rxq_ptr;
  77. } VIOsPAPRVLANDevice;
  78. static int spapr_vlan_can_receive(NetClientState *nc)
  79. {
  80. VIOsPAPRVLANDevice *dev = qemu_get_nic_opaque(nc);
  81. return (dev->isopen && dev->rx_bufs > 0);
  82. }
  83. static ssize_t spapr_vlan_receive(NetClientState *nc, const uint8_t *buf,
  84. size_t size)
  85. {
  86. VIOsPAPRVLANDevice *dev = qemu_get_nic_opaque(nc);
  87. VIOsPAPRDevice *sdev = VIO_SPAPR_DEVICE(dev);
  88. vlan_bd_t rxq_bd = vio_ldq(sdev, dev->buf_list + VLAN_RXQ_BD_OFF);
  89. vlan_bd_t bd;
  90. int buf_ptr = dev->use_buf_ptr;
  91. uint64_t handle;
  92. uint8_t control;
  93. DPRINTF("spapr_vlan_receive() [%s] rx_bufs=%d\n", sdev->qdev.id,
  94. dev->rx_bufs);
  95. if (!dev->isopen) {
  96. return -1;
  97. }
  98. if (!dev->rx_bufs) {
  99. return -1;
  100. }
  101. do {
  102. buf_ptr += 8;
  103. if (buf_ptr >= SPAPR_TCE_PAGE_SIZE) {
  104. buf_ptr = VLAN_RX_BDS_OFF;
  105. }
  106. bd = vio_ldq(sdev, dev->buf_list + buf_ptr);
  107. DPRINTF("use_buf_ptr=%d bd=0x%016llx\n",
  108. buf_ptr, (unsigned long long)bd);
  109. } while ((!(bd & VLAN_BD_VALID) || (VLAN_BD_LEN(bd) < (size + 8)))
  110. && (buf_ptr != dev->use_buf_ptr));
  111. if (!(bd & VLAN_BD_VALID) || (VLAN_BD_LEN(bd) < (size + 8))) {
  112. /* Failed to find a suitable buffer */
  113. return -1;
  114. }
  115. /* Remove the buffer from the pool */
  116. dev->rx_bufs--;
  117. dev->use_buf_ptr = buf_ptr;
  118. vio_stq(sdev, dev->buf_list + dev->use_buf_ptr, 0);
  119. DPRINTF("Found buffer: ptr=%d num=%d\n", dev->use_buf_ptr, dev->rx_bufs);
  120. /* Transfer the packet data */
  121. if (spapr_vio_dma_write(sdev, VLAN_BD_ADDR(bd) + 8, buf, size) < 0) {
  122. return -1;
  123. }
  124. DPRINTF("spapr_vlan_receive: DMA write completed\n");
  125. /* Update the receive queue */
  126. control = VLAN_RXQC_TOGGLE | VLAN_RXQC_VALID;
  127. if (rxq_bd & VLAN_BD_TOGGLE) {
  128. control ^= VLAN_RXQC_TOGGLE;
  129. }
  130. handle = vio_ldq(sdev, VLAN_BD_ADDR(bd));
  131. vio_stq(sdev, VLAN_BD_ADDR(rxq_bd) + dev->rxq_ptr + 8, handle);
  132. vio_stl(sdev, VLAN_BD_ADDR(rxq_bd) + dev->rxq_ptr + 4, size);
  133. vio_sth(sdev, VLAN_BD_ADDR(rxq_bd) + dev->rxq_ptr + 2, 8);
  134. vio_stb(sdev, VLAN_BD_ADDR(rxq_bd) + dev->rxq_ptr, control);
  135. DPRINTF("wrote rxq entry (ptr=0x%llx): 0x%016llx 0x%016llx\n",
  136. (unsigned long long)dev->rxq_ptr,
  137. (unsigned long long)vio_ldq(sdev, VLAN_BD_ADDR(rxq_bd) +
  138. dev->rxq_ptr),
  139. (unsigned long long)vio_ldq(sdev, VLAN_BD_ADDR(rxq_bd) +
  140. dev->rxq_ptr + 8));
  141. dev->rxq_ptr += 16;
  142. if (dev->rxq_ptr >= VLAN_BD_LEN(rxq_bd)) {
  143. dev->rxq_ptr = 0;
  144. vio_stq(sdev, dev->buf_list + VLAN_RXQ_BD_OFF, rxq_bd ^ VLAN_BD_TOGGLE);
  145. }
  146. if (sdev->signal_state & 1) {
  147. qemu_irq_pulse(spapr_vio_qirq(sdev));
  148. }
  149. return size;
  150. }
  151. static void spapr_vlan_cleanup(NetClientState *nc)
  152. {
  153. VIOsPAPRVLANDevice *dev = qemu_get_nic_opaque(nc);
  154. dev->nic = NULL;
  155. }
  156. static NetClientInfo net_spapr_vlan_info = {
  157. .type = NET_CLIENT_OPTIONS_KIND_NIC,
  158. .size = sizeof(NICState),
  159. .can_receive = spapr_vlan_can_receive,
  160. .receive = spapr_vlan_receive,
  161. .cleanup = spapr_vlan_cleanup,
  162. };
  163. static void spapr_vlan_reset(VIOsPAPRDevice *sdev)
  164. {
  165. VIOsPAPRVLANDevice *dev = VIO_SPAPR_VLAN_DEVICE(sdev);
  166. dev->buf_list = 0;
  167. dev->rx_bufs = 0;
  168. dev->isopen = 0;
  169. }
  170. static int spapr_vlan_init(VIOsPAPRDevice *sdev)
  171. {
  172. VIOsPAPRVLANDevice *dev = VIO_SPAPR_VLAN_DEVICE(sdev);
  173. qemu_macaddr_default_if_unset(&dev->nicconf.macaddr);
  174. dev->nic = qemu_new_nic(&net_spapr_vlan_info, &dev->nicconf,
  175. object_get_typename(OBJECT(sdev)), sdev->qdev.id, dev);
  176. qemu_format_nic_info_str(qemu_get_queue(dev->nic), dev->nicconf.macaddr.a);
  177. add_boot_device_path(dev->nicconf.bootindex, DEVICE(dev), "");
  178. return 0;
  179. }
  180. void spapr_vlan_create(VIOsPAPRBus *bus, NICInfo *nd)
  181. {
  182. DeviceState *dev;
  183. dev = qdev_create(&bus->bus, "spapr-vlan");
  184. qdev_set_nic_properties(dev, nd);
  185. qdev_init_nofail(dev);
  186. }
  187. static int spapr_vlan_devnode(VIOsPAPRDevice *dev, void *fdt, int node_off)
  188. {
  189. VIOsPAPRVLANDevice *vdev = VIO_SPAPR_VLAN_DEVICE(dev);
  190. uint8_t padded_mac[8] = {0, 0};
  191. int ret;
  192. /* Some old phyp versions give the mac address in an 8-byte
  193. * property. The kernel driver has an insane workaround for this;
  194. * rather than doing the obvious thing and checking the property
  195. * length, it checks whether the first byte has 0b10 in the low
  196. * bits. If a correct 6-byte property has a different first byte
  197. * the kernel will get the wrong mac address, overrunning its
  198. * buffer in the process (read only, thank goodness).
  199. *
  200. * Here we workaround the kernel workaround by always supplying an
  201. * 8-byte property, with the mac address in the last six bytes */
  202. memcpy(&padded_mac[2], &vdev->nicconf.macaddr, ETH_ALEN);
  203. ret = fdt_setprop(fdt, node_off, "local-mac-address",
  204. padded_mac, sizeof(padded_mac));
  205. if (ret < 0) {
  206. return ret;
  207. }
  208. ret = fdt_setprop_cell(fdt, node_off, "ibm,mac-address-filters", 0);
  209. if (ret < 0) {
  210. return ret;
  211. }
  212. return 0;
  213. }
  214. static int check_bd(VIOsPAPRVLANDevice *dev, vlan_bd_t bd,
  215. target_ulong alignment)
  216. {
  217. if ((VLAN_BD_ADDR(bd) % alignment)
  218. || (VLAN_BD_LEN(bd) % alignment)) {
  219. return -1;
  220. }
  221. if (!spapr_vio_dma_valid(&dev->sdev, VLAN_BD_ADDR(bd),
  222. VLAN_BD_LEN(bd), DMA_DIRECTION_FROM_DEVICE)
  223. || !spapr_vio_dma_valid(&dev->sdev, VLAN_BD_ADDR(bd),
  224. VLAN_BD_LEN(bd), DMA_DIRECTION_TO_DEVICE)) {
  225. return -1;
  226. }
  227. return 0;
  228. }
  229. static target_ulong h_register_logical_lan(PowerPCCPU *cpu,
  230. sPAPREnvironment *spapr,
  231. target_ulong opcode,
  232. target_ulong *args)
  233. {
  234. target_ulong reg = args[0];
  235. target_ulong buf_list = args[1];
  236. target_ulong rec_queue = args[2];
  237. target_ulong filter_list = args[3];
  238. VIOsPAPRDevice *sdev = spapr_vio_find_by_reg(spapr->vio_bus, reg);
  239. VIOsPAPRVLANDevice *dev = VIO_SPAPR_VLAN_DEVICE(sdev);
  240. vlan_bd_t filter_list_bd;
  241. if (!dev) {
  242. return H_PARAMETER;
  243. }
  244. if (dev->isopen) {
  245. hcall_dprintf("H_REGISTER_LOGICAL_LAN called twice without "
  246. "H_FREE_LOGICAL_LAN\n");
  247. return H_RESOURCE;
  248. }
  249. if (check_bd(dev, VLAN_VALID_BD(buf_list, SPAPR_TCE_PAGE_SIZE),
  250. SPAPR_TCE_PAGE_SIZE) < 0) {
  251. hcall_dprintf("Bad buf_list 0x" TARGET_FMT_lx "\n", buf_list);
  252. return H_PARAMETER;
  253. }
  254. filter_list_bd = VLAN_VALID_BD(filter_list, SPAPR_TCE_PAGE_SIZE);
  255. if (check_bd(dev, filter_list_bd, SPAPR_TCE_PAGE_SIZE) < 0) {
  256. hcall_dprintf("Bad filter_list 0x" TARGET_FMT_lx "\n", filter_list);
  257. return H_PARAMETER;
  258. }
  259. if (!(rec_queue & VLAN_BD_VALID)
  260. || (check_bd(dev, rec_queue, VLAN_RQ_ALIGNMENT) < 0)) {
  261. hcall_dprintf("Bad receive queue\n");
  262. return H_PARAMETER;
  263. }
  264. dev->buf_list = buf_list;
  265. sdev->signal_state = 0;
  266. rec_queue &= ~VLAN_BD_TOGGLE;
  267. /* Initialize the buffer list */
  268. vio_stq(sdev, buf_list, rec_queue);
  269. vio_stq(sdev, buf_list + 8, filter_list_bd);
  270. spapr_vio_dma_set(sdev, buf_list + VLAN_RX_BDS_OFF, 0,
  271. SPAPR_TCE_PAGE_SIZE - VLAN_RX_BDS_OFF);
  272. dev->add_buf_ptr = VLAN_RX_BDS_OFF - 8;
  273. dev->use_buf_ptr = VLAN_RX_BDS_OFF - 8;
  274. dev->rx_bufs = 0;
  275. dev->rxq_ptr = 0;
  276. /* Initialize the receive queue */
  277. spapr_vio_dma_set(sdev, VLAN_BD_ADDR(rec_queue), 0, VLAN_BD_LEN(rec_queue));
  278. dev->isopen = 1;
  279. qemu_flush_queued_packets(qemu_get_queue(dev->nic));
  280. return H_SUCCESS;
  281. }
  282. static target_ulong h_free_logical_lan(PowerPCCPU *cpu, sPAPREnvironment *spapr,
  283. target_ulong opcode, target_ulong *args)
  284. {
  285. target_ulong reg = args[0];
  286. VIOsPAPRDevice *sdev = spapr_vio_find_by_reg(spapr->vio_bus, reg);
  287. VIOsPAPRVLANDevice *dev = VIO_SPAPR_VLAN_DEVICE(sdev);
  288. if (!dev) {
  289. return H_PARAMETER;
  290. }
  291. if (!dev->isopen) {
  292. hcall_dprintf("H_FREE_LOGICAL_LAN called without "
  293. "H_REGISTER_LOGICAL_LAN\n");
  294. return H_RESOURCE;
  295. }
  296. spapr_vlan_reset(sdev);
  297. return H_SUCCESS;
  298. }
  299. static target_ulong h_add_logical_lan_buffer(PowerPCCPU *cpu,
  300. sPAPREnvironment *spapr,
  301. target_ulong opcode,
  302. target_ulong *args)
  303. {
  304. target_ulong reg = args[0];
  305. target_ulong buf = args[1];
  306. VIOsPAPRDevice *sdev = spapr_vio_find_by_reg(spapr->vio_bus, reg);
  307. VIOsPAPRVLANDevice *dev = VIO_SPAPR_VLAN_DEVICE(sdev);
  308. vlan_bd_t bd;
  309. DPRINTF("H_ADD_LOGICAL_LAN_BUFFER(0x" TARGET_FMT_lx
  310. ", 0x" TARGET_FMT_lx ")\n", reg, buf);
  311. if (!sdev) {
  312. hcall_dprintf("Bad device\n");
  313. return H_PARAMETER;
  314. }
  315. if ((check_bd(dev, buf, 4) < 0)
  316. || (VLAN_BD_LEN(buf) < 16)) {
  317. hcall_dprintf("Bad buffer enqueued\n");
  318. return H_PARAMETER;
  319. }
  320. if (!dev->isopen || dev->rx_bufs >= VLAN_MAX_BUFS) {
  321. return H_RESOURCE;
  322. }
  323. do {
  324. dev->add_buf_ptr += 8;
  325. if (dev->add_buf_ptr >= SPAPR_TCE_PAGE_SIZE) {
  326. dev->add_buf_ptr = VLAN_RX_BDS_OFF;
  327. }
  328. bd = vio_ldq(sdev, dev->buf_list + dev->add_buf_ptr);
  329. } while (bd & VLAN_BD_VALID);
  330. vio_stq(sdev, dev->buf_list + dev->add_buf_ptr, buf);
  331. dev->rx_bufs++;
  332. qemu_flush_queued_packets(qemu_get_queue(dev->nic));
  333. DPRINTF("h_add_logical_lan_buffer(): Added buf ptr=%d rx_bufs=%d"
  334. " bd=0x%016llx\n", dev->add_buf_ptr, dev->rx_bufs,
  335. (unsigned long long)buf);
  336. return H_SUCCESS;
  337. }
  338. static target_ulong h_send_logical_lan(PowerPCCPU *cpu, sPAPREnvironment *spapr,
  339. target_ulong opcode, target_ulong *args)
  340. {
  341. target_ulong reg = args[0];
  342. target_ulong *bufs = args + 1;
  343. target_ulong continue_token = args[7];
  344. VIOsPAPRDevice *sdev = spapr_vio_find_by_reg(spapr->vio_bus, reg);
  345. VIOsPAPRVLANDevice *dev = VIO_SPAPR_VLAN_DEVICE(sdev);
  346. unsigned total_len;
  347. uint8_t *lbuf, *p;
  348. int i, nbufs;
  349. int ret;
  350. DPRINTF("H_SEND_LOGICAL_LAN(0x" TARGET_FMT_lx ", <bufs>, 0x"
  351. TARGET_FMT_lx ")\n", reg, continue_token);
  352. if (!sdev) {
  353. return H_PARAMETER;
  354. }
  355. DPRINTF("rxbufs = %d\n", dev->rx_bufs);
  356. if (!dev->isopen) {
  357. return H_DROPPED;
  358. }
  359. if (continue_token) {
  360. return H_HARDWARE; /* FIXME actually handle this */
  361. }
  362. total_len = 0;
  363. for (i = 0; i < 6; i++) {
  364. DPRINTF(" buf desc: 0x" TARGET_FMT_lx "\n", bufs[i]);
  365. if (!(bufs[i] & VLAN_BD_VALID)) {
  366. break;
  367. }
  368. total_len += VLAN_BD_LEN(bufs[i]);
  369. }
  370. nbufs = i;
  371. DPRINTF("h_send_logical_lan() %d buffers, total length 0x%x\n",
  372. nbufs, total_len);
  373. if (total_len == 0) {
  374. return H_SUCCESS;
  375. }
  376. if (total_len > MAX_PACKET_SIZE) {
  377. /* Don't let the guest force too large an allocation */
  378. return H_RESOURCE;
  379. }
  380. lbuf = alloca(total_len);
  381. p = lbuf;
  382. for (i = 0; i < nbufs; i++) {
  383. ret = spapr_vio_dma_read(sdev, VLAN_BD_ADDR(bufs[i]),
  384. p, VLAN_BD_LEN(bufs[i]));
  385. if (ret < 0) {
  386. return ret;
  387. }
  388. p += VLAN_BD_LEN(bufs[i]);
  389. }
  390. qemu_send_packet(qemu_get_queue(dev->nic), lbuf, total_len);
  391. return H_SUCCESS;
  392. }
  393. static target_ulong h_multicast_ctrl(PowerPCCPU *cpu, sPAPREnvironment *spapr,
  394. target_ulong opcode, target_ulong *args)
  395. {
  396. target_ulong reg = args[0];
  397. VIOsPAPRDevice *dev = spapr_vio_find_by_reg(spapr->vio_bus, reg);
  398. if (!dev) {
  399. return H_PARAMETER;
  400. }
  401. return H_SUCCESS;
  402. }
  403. static Property spapr_vlan_properties[] = {
  404. DEFINE_SPAPR_PROPERTIES(VIOsPAPRVLANDevice, sdev),
  405. DEFINE_NIC_PROPERTIES(VIOsPAPRVLANDevice, nicconf),
  406. DEFINE_PROP_END_OF_LIST(),
  407. };
  408. static const VMStateDescription vmstate_spapr_llan = {
  409. .name = "spapr_llan",
  410. .version_id = 1,
  411. .minimum_version_id = 1,
  412. .minimum_version_id_old = 1,
  413. .fields = (VMStateField []) {
  414. VMSTATE_SPAPR_VIO(sdev, VIOsPAPRVLANDevice),
  415. /* LLAN state */
  416. VMSTATE_BOOL(isopen, VIOsPAPRVLANDevice),
  417. VMSTATE_UINTTL(buf_list, VIOsPAPRVLANDevice),
  418. VMSTATE_UINT32(add_buf_ptr, VIOsPAPRVLANDevice),
  419. VMSTATE_UINT32(use_buf_ptr, VIOsPAPRVLANDevice),
  420. VMSTATE_UINT32(rx_bufs, VIOsPAPRVLANDevice),
  421. VMSTATE_UINTTL(rxq_ptr, VIOsPAPRVLANDevice),
  422. VMSTATE_END_OF_LIST()
  423. },
  424. };
  425. static void spapr_vlan_class_init(ObjectClass *klass, void *data)
  426. {
  427. DeviceClass *dc = DEVICE_CLASS(klass);
  428. VIOsPAPRDeviceClass *k = VIO_SPAPR_DEVICE_CLASS(klass);
  429. k->init = spapr_vlan_init;
  430. k->reset = spapr_vlan_reset;
  431. k->devnode = spapr_vlan_devnode;
  432. k->dt_name = "l-lan";
  433. k->dt_type = "network";
  434. k->dt_compatible = "IBM,l-lan";
  435. k->signal_mask = 0x1;
  436. set_bit(DEVICE_CATEGORY_NETWORK, dc->categories);
  437. dc->props = spapr_vlan_properties;
  438. k->rtce_window_size = 0x10000000;
  439. dc->vmsd = &vmstate_spapr_llan;
  440. }
  441. static const TypeInfo spapr_vlan_info = {
  442. .name = TYPE_VIO_SPAPR_VLAN_DEVICE,
  443. .parent = TYPE_VIO_SPAPR_DEVICE,
  444. .instance_size = sizeof(VIOsPAPRVLANDevice),
  445. .class_init = spapr_vlan_class_init,
  446. };
  447. static void spapr_vlan_register_types(void)
  448. {
  449. spapr_register_hypercall(H_REGISTER_LOGICAL_LAN, h_register_logical_lan);
  450. spapr_register_hypercall(H_FREE_LOGICAL_LAN, h_free_logical_lan);
  451. spapr_register_hypercall(H_SEND_LOGICAL_LAN, h_send_logical_lan);
  452. spapr_register_hypercall(H_ADD_LOGICAL_LAN_BUFFER,
  453. h_add_logical_lan_buffer);
  454. spapr_register_hypercall(H_MULTICAST_CTRL, h_multicast_ctrl);
  455. type_register_static(&spapr_vlan_info);
  456. }
  457. type_init(spapr_vlan_register_types)