xen_devconfig.c 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174
  1. #include "xen_backend.h"
  2. #include "blockdev.h"
  3. /* ------------------------------------------------------------- */
  4. struct xs_dirs {
  5. char *xs_dir;
  6. QTAILQ_ENTRY(xs_dirs) list;
  7. };
  8. static QTAILQ_HEAD(xs_dirs_head, xs_dirs) xs_cleanup = QTAILQ_HEAD_INITIALIZER(xs_cleanup);
  9. static void xen_config_cleanup_dir(char *dir)
  10. {
  11. struct xs_dirs *d;
  12. d = g_malloc(sizeof(*d));
  13. d->xs_dir = dir;
  14. QTAILQ_INSERT_TAIL(&xs_cleanup, d, list);
  15. }
  16. void xen_config_cleanup(void)
  17. {
  18. struct xs_dirs *d;
  19. QTAILQ_FOREACH(d, &xs_cleanup, list) {
  20. xs_rm(xenstore, 0, d->xs_dir);
  21. }
  22. }
  23. /* ------------------------------------------------------------- */
  24. static int xen_config_dev_mkdir(char *dev, int p)
  25. {
  26. struct xs_permissions perms[2] = {{
  27. .id = 0, /* set owner: dom0 */
  28. },{
  29. .id = xen_domid,
  30. .perms = p,
  31. }};
  32. if (!xs_mkdir(xenstore, 0, dev)) {
  33. xen_be_printf(NULL, 0, "xs_mkdir %s: failed\n", dev);
  34. return -1;
  35. }
  36. xen_config_cleanup_dir(g_strdup(dev));
  37. if (!xs_set_permissions(xenstore, 0, dev, perms, 2)) {
  38. xen_be_printf(NULL, 0, "xs_set_permissions %s: failed\n", dev);
  39. return -1;
  40. }
  41. return 0;
  42. }
  43. static int xen_config_dev_dirs(const char *ftype, const char *btype, int vdev,
  44. char *fe, char *be, int len)
  45. {
  46. char *dom;
  47. dom = xs_get_domain_path(xenstore, xen_domid);
  48. snprintf(fe, len, "%s/device/%s/%d", dom, ftype, vdev);
  49. free(dom);
  50. dom = xs_get_domain_path(xenstore, 0);
  51. snprintf(be, len, "%s/backend/%s/%d/%d", dom, btype, xen_domid, vdev);
  52. free(dom);
  53. xen_config_dev_mkdir(fe, XS_PERM_READ | XS_PERM_WRITE);
  54. xen_config_dev_mkdir(be, XS_PERM_READ);
  55. return 0;
  56. }
  57. static int xen_config_dev_all(char *fe, char *be)
  58. {
  59. /* frontend */
  60. if (xen_protocol)
  61. xenstore_write_str(fe, "protocol", xen_protocol);
  62. xenstore_write_int(fe, "state", XenbusStateInitialising);
  63. xenstore_write_int(fe, "backend-id", 0);
  64. xenstore_write_str(fe, "backend", be);
  65. /* backend */
  66. xenstore_write_str(be, "domain", qemu_name ? qemu_name : "no-name");
  67. xenstore_write_int(be, "online", 1);
  68. xenstore_write_int(be, "state", XenbusStateInitialising);
  69. xenstore_write_int(be, "frontend-id", xen_domid);
  70. xenstore_write_str(be, "frontend", fe);
  71. return 0;
  72. }
  73. /* ------------------------------------------------------------- */
  74. int xen_config_dev_blk(DriveInfo *disk)
  75. {
  76. char fe[256], be[256], device_name[32];
  77. int vdev = 202 * 256 + 16 * disk->unit;
  78. int cdrom = disk->media_cd;
  79. const char *devtype = cdrom ? "cdrom" : "disk";
  80. const char *mode = cdrom ? "r" : "w";
  81. const char *filename = qemu_opt_get(disk->opts, "file");
  82. snprintf(device_name, sizeof(device_name), "xvd%c", 'a' + disk->unit);
  83. xen_be_printf(NULL, 1, "config disk %d [%s]: %s\n",
  84. disk->unit, device_name, filename);
  85. xen_config_dev_dirs("vbd", "qdisk", vdev, fe, be, sizeof(fe));
  86. /* frontend */
  87. xenstore_write_int(fe, "virtual-device", vdev);
  88. xenstore_write_str(fe, "device-type", devtype);
  89. /* backend */
  90. xenstore_write_str(be, "dev", device_name);
  91. xenstore_write_str(be, "type", "file");
  92. xenstore_write_str(be, "params", filename);
  93. xenstore_write_str(be, "mode", mode);
  94. /* common stuff */
  95. return xen_config_dev_all(fe, be);
  96. }
  97. int xen_config_dev_nic(NICInfo *nic)
  98. {
  99. char fe[256], be[256];
  100. char mac[20];
  101. int vlan_id = -1;
  102. net_hub_id_for_client(nic->netdev, &vlan_id);
  103. snprintf(mac, sizeof(mac), "%02x:%02x:%02x:%02x:%02x:%02x",
  104. nic->macaddr.a[0], nic->macaddr.a[1], nic->macaddr.a[2],
  105. nic->macaddr.a[3], nic->macaddr.a[4], nic->macaddr.a[5]);
  106. xen_be_printf(NULL, 1, "config nic %d: mac=\"%s\"\n", vlan_id, mac);
  107. xen_config_dev_dirs("vif", "qnic", vlan_id, fe, be, sizeof(fe));
  108. /* frontend */
  109. xenstore_write_int(fe, "handle", vlan_id);
  110. xenstore_write_str(fe, "mac", mac);
  111. /* backend */
  112. xenstore_write_int(be, "handle", vlan_id);
  113. xenstore_write_str(be, "mac", mac);
  114. /* common stuff */
  115. return xen_config_dev_all(fe, be);
  116. }
  117. int xen_config_dev_vfb(int vdev, const char *type)
  118. {
  119. char fe[256], be[256];
  120. xen_config_dev_dirs("vfb", "vfb", vdev, fe, be, sizeof(fe));
  121. /* backend */
  122. xenstore_write_str(be, "type", type);
  123. /* common stuff */
  124. return xen_config_dev_all(fe, be);
  125. }
  126. int xen_config_dev_vkbd(int vdev)
  127. {
  128. char fe[256], be[256];
  129. xen_config_dev_dirs("vkbd", "vkbd", vdev, fe, be, sizeof(fe));
  130. return xen_config_dev_all(fe, be);
  131. }
  132. int xen_config_dev_console(int vdev)
  133. {
  134. char fe[256], be[256];
  135. xen_config_dev_dirs("console", "console", vdev, fe, be, sizeof(fe));
  136. return xen_config_dev_all(fe, be);
  137. }