| 
									
										
										
										
											2019-06-20 18:37:08 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  *  qdev vm change state handlers | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  This program is free software; you can redistribute it and/or modify | 
					
						
							|  |  |  |  *  it under the terms of the GNU General Public License as published by | 
					
						
							|  |  |  |  *  the Free Software Foundation; either version 2 of the License, | 
					
						
							|  |  |  |  *  or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  *  but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  *  GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  *  along with this program; if not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "qemu/osdep.h"
 | 
					
						
							| 
									
										
										
										
											2019-08-12 07:23:51 +02:00
										 |  |  | #include "hw/qdev-core.h"
 | 
					
						
							| 
									
										
										
										
											2019-08-12 07:23:59 +02:00
										 |  |  | #include "sysemu/runstate.h"
 | 
					
						
							| 
									
										
										
										
											2019-06-20 18:37:08 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | static int qdev_get_dev_tree_depth(DeviceState *dev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int depth; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (depth = 0; dev; depth++) { | 
					
						
							|  |  |  |         BusState *bus = dev->parent_bus; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!bus) { | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         dev = bus->parent; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return depth; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * qdev_add_vm_change_state_handler: | 
					
						
							|  |  |  |  * @dev: the device that owns this handler | 
					
						
							|  |  |  |  * @cb: the callback function to be invoked | 
					
						
							|  |  |  |  * @opaque: user data passed to the callback function | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function works like qemu_add_vm_change_state_handler() except callbacks | 
					
						
							|  |  |  |  * are invoked in qdev tree depth order.  Ordering is desirable when callbacks | 
					
						
							|  |  |  |  * of children depend on their parent's callback having completed first. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * For example, when qdev_add_vm_change_state_handler() is used, a host | 
					
						
							|  |  |  |  * controller's callback is invoked before the children on its bus when the VM | 
					
						
							|  |  |  |  * starts running.  The order is reversed when the VM stops running. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Returns: an entry to be freed with qemu_del_vm_change_state_handler() | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | VMChangeStateEntry *qdev_add_vm_change_state_handler(DeviceState *dev, | 
					
						
							|  |  |  |                                                      VMChangeStateHandler *cb, | 
					
						
							|  |  |  |                                                      void *opaque) | 
					
						
							| 
									
										
										
										
											2023-08-02 11:14:46 +03:00
										 |  |  | { | 
					
						
							|  |  |  |     return qdev_add_vm_change_state_handler_full(dev, cb, NULL, opaque); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Exactly like qdev_add_vm_change_state_handler() but passes a prepare_cb | 
					
						
							|  |  |  |  * argument too. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | VMChangeStateEntry *qdev_add_vm_change_state_handler_full( | 
					
						
							|  |  |  |     DeviceState *dev, VMChangeStateHandler *cb, | 
					
						
							|  |  |  |     VMChangeStateHandler *prepare_cb, void *opaque) | 
					
						
							| 
									
										
										
										
											2019-06-20 18:37:08 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     int depth = qdev_get_dev_tree_depth(dev); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-02 11:14:46 +03:00
										 |  |  |     return qemu_add_vm_change_state_handler_prio_full(cb, prepare_cb, opaque, | 
					
						
							|  |  |  |                                                       depth); | 
					
						
							| 
									
										
										
										
											2019-06-20 18:37:08 +01:00
										 |  |  | } |