| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:45 +10:00
										 |  |  |  * Deprecated PCI hotplug interface support | 
					
						
							|  |  |  |  * This covers the old pci_add / pci_del command, whereas the more general | 
					
						
							|  |  |  |  * device_add / device_del commands are now preferred. | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Copyright (c) 2004 Fabrice Bellard | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Permission is hereby granted, free of charge, to any person obtaining a copy | 
					
						
							|  |  |  |  * of this software and associated documentation files (the "Software"), to deal | 
					
						
							|  |  |  |  * in the Software without restriction, including without limitation the rights | 
					
						
							|  |  |  |  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | 
					
						
							|  |  |  |  * copies of the Software, and to permit persons to whom the Software is | 
					
						
							|  |  |  |  * furnished to do so, subject to the following conditions: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The above copyright notice and this permission notice shall be included in | 
					
						
							|  |  |  |  * all copies or substantial portions of the Software. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | 
					
						
							|  |  |  |  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
					
						
							|  |  |  |  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL | 
					
						
							|  |  |  |  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | 
					
						
							|  |  |  |  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | 
					
						
							|  |  |  |  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | 
					
						
							|  |  |  |  * THE SOFTWARE. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-12 23:05:42 +02:00
										 |  |  | #include "hw/hw.h"
 | 
					
						
							|  |  |  | #include "hw/boards.h"
 | 
					
						
							|  |  |  | #include "hw/pci/pci.h"
 | 
					
						
							| 
									
										
										
										
											2012-10-24 08:43:34 +02:00
										 |  |  | #include "net/net.h"
 | 
					
						
							| 
									
										
										
										
											2013-02-05 17:06:20 +01:00
										 |  |  | #include "hw/i386/pc.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:19:49 +01:00
										 |  |  | #include "monitor/monitor.h"
 | 
					
						
							| 
									
										
										
										
											2013-02-05 17:06:20 +01:00
										 |  |  | #include "hw/scsi/scsi.h"
 | 
					
						
							|  |  |  | #include "hw/virtio/virtio-blk.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:20:00 +01:00
										 |  |  | #include "qemu/config-file.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:20:04 +01:00
										 |  |  | #include "sysemu/blockdev.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:19:43 +01:00
										 |  |  | #include "qapi/error.h"
 | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:47 +10:00
										 |  |  | static int pci_read_devaddr(Monitor *mon, const char *addr, | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:46 +10:00
										 |  |  |                             int *busp, unsigned *slotp) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:47 +10:00
										 |  |  |     int dom; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:46 +10:00
										 |  |  |     /* strip legacy tag */ | 
					
						
							|  |  |  |     if (!strncmp(addr, "pci_addr=", 9)) { | 
					
						
							|  |  |  |         addr += 9; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:47 +10:00
										 |  |  |     if (pci_parse_devaddr(addr, &dom, busp, slotp, NULL)) { | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:46 +10:00
										 |  |  |         monitor_printf(mon, "Invalid pci address\n"); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:47 +10:00
										 |  |  |     if (dom != 0) { | 
					
						
							|  |  |  |         monitor_printf(mon, "Multiple PCI domains not supported, use device_add\n"); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:46 +10:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-18 15:14:09 +02:00
										 |  |  | static PCIDevice *qemu_pci_hot_add_nic(Monitor *mon, | 
					
						
							| 
									
										
										
										
											2009-10-06 12:17:15 +01:00
										 |  |  |                                        const char *devaddr, | 
					
						
							|  |  |  |                                        const char *opts_str) | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-04-20 16:50:25 -03:00
										 |  |  |     Error *local_err = NULL; | 
					
						
							| 
									
										
										
										
											2009-10-06 12:17:15 +01:00
										 |  |  |     QemuOpts *opts; | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:52 +10:00
										 |  |  |     PCIBus *root = pci_find_primary_bus(); | 
					
						
							| 
									
										
										
										
											2009-12-10 11:11:05 +01:00
										 |  |  |     PCIBus *bus; | 
					
						
							|  |  |  |     int ret, devfn; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:52 +10:00
										 |  |  |     if (!root) { | 
					
						
							|  |  |  |         monitor_printf(mon, "no primary PCI bus (if there are multiple" | 
					
						
							|  |  |  |                        " PCI roots, you must use device_add instead)"); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bus = pci_get_bus_devfn(&devfn, root, devaddr); | 
					
						
							| 
									
										
										
										
											2009-12-10 11:11:05 +01:00
										 |  |  |     if (!bus) { | 
					
						
							|  |  |  |         monitor_printf(mon, "Invalid PCI device address %s\n", devaddr); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!((BusState*)bus)->allow_hotplug) { | 
					
						
							|  |  |  |         monitor_printf(mon, "PCI bus doesn't support hotplug\n"); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-20 13:52:01 +02:00
										 |  |  |     opts = qemu_opts_parse(qemu_find_opts("net"), opts_str ? opts_str : "", 0); | 
					
						
							| 
									
										
										
										
											2009-10-06 12:17:15 +01:00
										 |  |  |     if (!opts) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     qemu_opt_set(opts, "type", "nic"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-20 16:50:25 -03:00
										 |  |  |     ret = net_client_init(opts, 0, &local_err); | 
					
						
							| 
									
										
										
										
											2014-01-30 15:07:28 +01:00
										 |  |  |     if (local_err) { | 
					
						
							| 
									
										
										
										
											2012-04-20 16:50:25 -03:00
										 |  |  |         qerror_report_err(local_err); | 
					
						
							|  |  |  |         error_free(local_err); | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2012-04-20 16:50:25 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-06-18 15:14:08 +02:00
										 |  |  |     if (nd_table[ret].devaddr) { | 
					
						
							|  |  |  |         monitor_printf(mon, "Parameter addr not supported\n"); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:52 +10:00
										 |  |  |     return pci_nic_init(&nd_table[ret], root, "rtl8139", devaddr); | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-18 18:41:52 +01:00
										 |  |  | static int scsi_hot_add(Monitor *mon, DeviceState *adapter, | 
					
						
							|  |  |  |                         DriveInfo *dinfo, int printinfo) | 
					
						
							| 
									
										
										
										
											2009-10-14 15:30:22 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     SCSIBus *scsibus; | 
					
						
							|  |  |  |     SCSIDevice *scsidev; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-23 16:56:18 +01:00
										 |  |  |     scsibus = (SCSIBus *) | 
					
						
							|  |  |  |         object_dynamic_cast(OBJECT(QLIST_FIRST(&adapter->child_bus)), | 
					
						
							|  |  |  |                             TYPE_SCSI_BUS); | 
					
						
							|  |  |  |     if (!scsibus) { | 
					
						
							|  |  |  | 	error_report("Device is not a SCSI adapter"); | 
					
						
							|  |  |  | 	return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-10-14 15:30:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /*
 | 
					
						
							|  |  |  |      * drive_init() tries to find a default for dinfo->unit.  Doesn't | 
					
						
							|  |  |  |      * work at all for hotplug though as we assign the device to a | 
					
						
							|  |  |  |      * specific bus instead of the first bus with spare scsi ids. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * Ditch the calculated value and reload from option string (if | 
					
						
							|  |  |  |      * specified). | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     dinfo->unit = qemu_opt_get_number(dinfo->opts, "unit", -1); | 
					
						
							| 
									
										
										
										
											2011-01-28 11:21:36 +01:00
										 |  |  |     dinfo->bus = scsibus->busnr; | 
					
						
							| 
									
										
										
										
											2011-11-18 16:32:00 +01:00
										 |  |  |     scsidev = scsi_bus_legacy_add_drive(scsibus, dinfo->bdrv, dinfo->unit, | 
					
						
							| 
									
										
										
										
											2013-07-21 12:16:34 +02:00
										 |  |  |                                         false, -1, NULL, NULL); | 
					
						
							| 
									
										
										
										
											2010-06-25 18:53:21 +02:00
										 |  |  |     if (!scsidev) { | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-12-10 11:11:07 +01:00
										 |  |  |     dinfo->unit = scsidev->id; | 
					
						
							| 
									
										
										
										
											2009-10-14 15:30:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (printinfo) | 
					
						
							| 
									
										
										
										
											2010-02-18 18:41:52 +01:00
										 |  |  |         monitor_printf(mon, "OK bus %d, unit %d\n", | 
					
						
							|  |  |  |                        scsibus->busnr, scsidev->id); | 
					
						
							| 
									
										
										
										
											2009-10-14 15:30:22 +02:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-22 15:16:36 +01:00
										 |  |  | int pci_drive_hot_add(Monitor *mon, const QDict *qdict, DriveInfo *dinfo) | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:47 +10:00
										 |  |  |     int pci_bus; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |     unsigned slot; | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:47 +10:00
										 |  |  |     PCIBus *root = pci_find_primary_bus(); | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |     PCIDevice *dev; | 
					
						
							| 
									
										
										
										
											2009-08-28 15:27:14 -03:00
										 |  |  |     const char *pci_addr = qdict_get_str(qdict, "pci_addr"); | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-22 15:16:36 +01:00
										 |  |  |     switch (dinfo->type) { | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |     case IF_SCSI: | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:47 +10:00
										 |  |  |         if (!root) { | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:52 +10:00
										 |  |  |             monitor_printf(mon, "no primary PCI bus (if there are multiple" | 
					
						
							|  |  |  |                            " PCI roots, you must use device_add instead)"); | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:47 +10:00
										 |  |  |             goto err; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (pci_read_devaddr(mon, pci_addr, &pci_bus, &slot)) { | 
					
						
							| 
									
										
										
										
											2009-09-25 21:42:47 +02:00
										 |  |  |             goto err; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:47 +10:00
										 |  |  |         dev = pci_find_device(root, pci_bus, PCI_DEVFN(slot, 0)); | 
					
						
							| 
									
										
										
										
											2009-09-25 21:42:47 +02:00
										 |  |  |         if (!dev) { | 
					
						
							|  |  |  |             monitor_printf(mon, "no pci device with address %s\n", pci_addr); | 
					
						
							|  |  |  |             goto err; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-02-18 18:41:52 +01:00
										 |  |  |         if (scsi_hot_add(mon, &dev->qdev, dinfo, 1) != 0) { | 
					
						
							| 
									
										
										
										
											2009-10-14 15:30:22 +02:00
										 |  |  |             goto err; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2012-11-22 15:16:36 +01:00
										 |  |  |         monitor_printf(mon, "Can't hot-add drive to type %d\n", dinfo->type); | 
					
						
							| 
									
										
										
										
											2009-09-25 21:42:47 +02:00
										 |  |  |         goto err; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-23 23:43:10 +02:00
										 |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2009-09-25 21:42:47 +02:00
										 |  |  | err: | 
					
						
							| 
									
										
										
										
											2010-08-23 23:43:10 +02:00
										 |  |  |     return -1; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-18 15:14:09 +02:00
										 |  |  | static PCIDevice *qemu_pci_hot_add_storage(Monitor *mon, | 
					
						
							|  |  |  |                                            const char *devaddr, | 
					
						
							| 
									
										
										
										
											2009-03-05 23:01:23 +00:00
										 |  |  |                                            const char *opts) | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-06-18 15:14:09 +02:00
										 |  |  |     PCIDevice *dev; | 
					
						
							| 
									
										
										
										
											2009-08-16 14:07:54 +02:00
										 |  |  |     DriveInfo *dinfo = NULL; | 
					
						
							| 
									
										
										
										
											2009-07-22 16:42:57 +02:00
										 |  |  |     int type = -1; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |     char buf[128]; | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:52 +10:00
										 |  |  |     PCIBus *root = pci_find_primary_bus(); | 
					
						
							| 
									
										
										
										
											2009-09-25 03:53:49 +02:00
										 |  |  |     PCIBus *bus; | 
					
						
							|  |  |  |     int devfn; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (get_param_value(buf, sizeof(buf), "if", opts)) { | 
					
						
							|  |  |  |         if (!strcmp(buf, "scsi")) | 
					
						
							|  |  |  |             type = IF_SCSI; | 
					
						
							|  |  |  |         else if (!strcmp(buf, "virtio")) { | 
					
						
							|  |  |  |             type = IF_VIRTIO; | 
					
						
							| 
									
										
										
										
											2009-04-05 17:40:55 +00:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             monitor_printf(mon, "type %s not a hotpluggable PCI device.\n", buf); | 
					
						
							| 
									
										
										
										
											2009-06-18 15:14:09 +02:00
										 |  |  |             return NULL; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2009-03-05 23:01:23 +00:00
										 |  |  |         monitor_printf(mon, "no if= specified\n"); | 
					
						
							| 
									
										
										
										
											2009-06-18 15:14:09 +02:00
										 |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (get_param_value(buf, sizeof(buf), "file", opts)) { | 
					
						
							| 
									
										
										
										
											2009-07-22 16:42:57 +02:00
										 |  |  |         dinfo = add_init_drive(opts); | 
					
						
							|  |  |  |         if (!dinfo) | 
					
						
							| 
									
										
										
										
											2009-06-18 15:14:09 +02:00
										 |  |  |             return NULL; | 
					
						
							| 
									
										
										
										
											2009-07-22 16:42:57 +02:00
										 |  |  |         if (dinfo->devaddr) { | 
					
						
							| 
									
										
										
										
											2009-06-18 15:14:10 +02:00
										 |  |  |             monitor_printf(mon, "Parameter addr not supported\n"); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-08-23 06:12:54 +00:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         dinfo = NULL; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:52 +10:00
										 |  |  |     if (!root) { | 
					
						
							|  |  |  |         monitor_printf(mon, "no primary PCI bus (if there are multiple" | 
					
						
							|  |  |  |                        " PCI roots, you must use device_add instead)"); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     bus = pci_get_bus_devfn(&devfn, root, devaddr); | 
					
						
							| 
									
										
										
										
											2009-09-25 03:53:49 +02:00
										 |  |  |     if (!bus) { | 
					
						
							|  |  |  |         monitor_printf(mon, "Invalid PCI device address %s\n", devaddr); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-12-10 11:11:05 +01:00
										 |  |  |     if (!((BusState*)bus)->allow_hotplug) { | 
					
						
							|  |  |  |         monitor_printf(mon, "PCI bus doesn't support hotplug\n"); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-09-25 03:53:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |     switch (type) { | 
					
						
							|  |  |  |     case IF_SCSI: | 
					
						
							| 
									
										
										
										
											2009-09-25 03:53:53 +02:00
										 |  |  |         dev = pci_create(bus, devfn, "lsi53c895a"); | 
					
						
							| 
									
										
										
										
											2009-10-13 13:59:55 +02:00
										 |  |  |         if (qdev_init(&dev->qdev) < 0) | 
					
						
							|  |  |  |             dev = NULL; | 
					
						
							| 
									
										
										
										
											2009-12-07 20:51:49 +00:00
										 |  |  |         if (dev && dinfo) { | 
					
						
							| 
									
										
										
										
											2010-02-18 18:41:52 +01:00
										 |  |  |             if (scsi_hot_add(mon, &dev->qdev, dinfo, 0) != 0) { | 
					
						
							| 
									
										
										
										
											2012-03-14 17:37:38 -03:00
										 |  |  |                 qdev_unplug(&dev->qdev, NULL); | 
					
						
							| 
									
										
										
										
											2009-10-14 15:30:22 +02:00
										 |  |  |                 dev = NULL; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-10-13 13:59:55 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case IF_VIRTIO: | 
					
						
							| 
									
										
										
										
											2009-08-23 06:12:54 +00:00
										 |  |  |         if (!dinfo) { | 
					
						
							|  |  |  |             monitor_printf(mon, "virtio requires a backing file/device.\n"); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-09-25 03:53:53 +02:00
										 |  |  |         dev = pci_create(bus, devfn, "virtio-blk-pci"); | 
					
						
							| 
									
										
										
										
											2010-06-29 16:58:30 +02:00
										 |  |  |         if (qdev_prop_set_drive(&dev->qdev, "drive", dinfo->bdrv) < 0) { | 
					
						
							| 
									
										
										
										
											2013-09-11 14:54:09 +02:00
										 |  |  |             object_unparent(OBJECT(dev)); | 
					
						
							| 
									
										
										
										
											2010-06-29 16:58:30 +02:00
										 |  |  |             dev = NULL; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-10-13 13:59:55 +02:00
										 |  |  |         if (qdev_init(&dev->qdev) < 0) | 
					
						
							|  |  |  |             dev = NULL; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2009-06-18 15:14:09 +02:00
										 |  |  |     default: | 
					
						
							|  |  |  |         dev = NULL; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-06-18 15:14:09 +02:00
										 |  |  |     return dev; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-12 10:53:00 +02:00
										 |  |  | void pci_device_hot_add(Monitor *mon, const QDict *qdict) | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     PCIDevice *dev = NULL; | 
					
						
							| 
									
										
										
										
											2009-08-28 15:27:15 -03:00
										 |  |  |     const char *pci_addr = qdict_get_str(qdict, "pci_addr"); | 
					
						
							|  |  |  |     const char *type = qdict_get_str(qdict, "type"); | 
					
						
							|  |  |  |     const char *opts = qdict_get_try_str(qdict, "opts"); | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-26 00:04:00 +02:00
										 |  |  |     /* strip legacy tag */ | 
					
						
							|  |  |  |     if (!strncmp(pci_addr, "pci_addr=", 9)) { | 
					
						
							|  |  |  |         pci_addr += 9; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-26 00:04:10 +02:00
										 |  |  |     if (!opts) { | 
					
						
							|  |  |  |         opts = ""; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-26 00:04:00 +02:00
										 |  |  |     if (!strcmp(pci_addr, "auto")) | 
					
						
							|  |  |  |         pci_addr = NULL; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-10 23:49:55 -02:00
										 |  |  |     if (strcmp(type, "nic") == 0) { | 
					
						
							| 
									
										
										
										
											2009-06-26 00:04:00 +02:00
										 |  |  |         dev = qemu_pci_hot_add_nic(mon, pci_addr, opts); | 
					
						
							| 
									
										
										
										
											2010-02-10 23:49:55 -02:00
										 |  |  |     } else if (strcmp(type, "storage") == 0) { | 
					
						
							| 
									
										
										
										
											2009-06-26 00:04:00 +02:00
										 |  |  |         dev = qemu_pci_hot_add_storage(mon, pci_addr, opts); | 
					
						
							| 
									
										
										
										
											2010-02-10 23:49:55 -02:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2009-03-05 23:01:23 +00:00
										 |  |  |         monitor_printf(mon, "invalid type: %s\n", type); | 
					
						
							| 
									
										
										
										
											2010-02-10 23:49:55 -02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (dev) { | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:49 +10:00
										 |  |  |         monitor_printf(mon, "OK root bus %s, bus %d, slot %d, function %d\n", | 
					
						
							|  |  |  |                        pci_root_bus_path(dev), | 
					
						
							| 
									
										
										
										
											2010-05-28 18:30:46 +09:00
										 |  |  |                        pci_bus_num(dev->bus), PCI_SLOT(dev->devfn), | 
					
						
							| 
									
										
										
										
											2010-05-12 10:53:00 +02:00
										 |  |  |                        PCI_FUNC(dev->devfn)); | 
					
						
							|  |  |  |     } else | 
					
						
							| 
									
										
										
										
											2009-03-05 23:01:23 +00:00
										 |  |  |         monitor_printf(mon, "failed to add %s\n", opts); | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-15 12:48:36 +09:00
										 |  |  | static int pci_device_hot_remove(Monitor *mon, const char *pci_addr) | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:47 +10:00
										 |  |  |     PCIBus *root = pci_find_primary_bus(); | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |     PCIDevice *d; | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:47 +10:00
										 |  |  |     int bus; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |     unsigned slot; | 
					
						
							| 
									
										
										
										
											2012-03-14 17:37:38 -03:00
										 |  |  |     Error *local_err = NULL; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:47 +10:00
										 |  |  |     if (!root) { | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:52 +10:00
										 |  |  |         monitor_printf(mon, "no primary PCI bus (if there are multiple" | 
					
						
							|  |  |  |                        " PCI roots, you must use device_del instead)"); | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:47 +10:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (pci_read_devaddr(mon, pci_addr, &bus, &slot)) { | 
					
						
							| 
									
										
										
										
											2010-02-10 23:49:56 -02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-06 18:48:47 +10:00
										 |  |  |     d = pci_find_device(root, bus, PCI_DEVFN(slot, 0)); | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |     if (!d) { | 
					
						
							| 
									
										
										
										
											2009-03-05 23:01:23 +00:00
										 |  |  |         monitor_printf(mon, "slot %d empty\n", slot); | 
					
						
							| 
									
										
										
										
											2010-02-10 23:49:56 -02:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-03-14 17:37:38 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     qdev_unplug(&d->qdev, &local_err); | 
					
						
							| 
									
										
										
										
											2014-01-30 15:07:28 +01:00
										 |  |  |     if (local_err) { | 
					
						
							| 
									
										
										
										
											2012-03-14 17:37:38 -03:00
										 |  |  |         monitor_printf(mon, "%s\n", error_get_pretty(local_err)); | 
					
						
							|  |  |  |         error_free(local_err); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2009-02-11 15:21:54 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-12 10:53:01 +02:00
										 |  |  | void do_pci_device_hot_remove(Monitor *mon, const QDict *qdict) | 
					
						
							| 
									
										
										
										
											2009-08-28 15:27:08 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-12 10:53:01 +02:00
										 |  |  |     pci_device_hot_remove(mon, qdict_get_str(qdict, "pci_addr")); | 
					
						
							| 
									
										
										
										
											2009-08-28 15:27:08 -03:00
										 |  |  | } |