 739fa32554
			
		
	
	
		739fa32554
		
	
	
	
	
		
			
			Clock canonical name is set in device_set_realized (see the block added to hw/core/qdev.c in commit 0e6934f264). If we connect a clock after the device is realized, this code is not executed. This is currently not a problem as this name is only used for trace events, however this disrupt tracing. Add a comment to document qdev_connect_clock_in() must be called before the device is realized, and assert this condition. Signed-off-by: Philippe Mathieu-Daudé <f4bug@amsat.org> Reviewed-by: Alistair Francis <alistair.francis@wdc.com> Message-id: 20200803105647.22223-5-f4bug@amsat.org Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
		
			
				
	
	
		
			158 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			158 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Device's clock input and output
 | |
|  *
 | |
|  * Copyright GreenSocs 2016-2020
 | |
|  *
 | |
|  * Authors:
 | |
|  *  Frederic Konrad
 | |
|  *  Damien Hedde
 | |
|  *
 | |
|  * This work is licensed under the terms of the GNU GPL, version 2 or later.
 | |
|  * See the COPYING file in the top-level directory.
 | |
|  */
 | |
| 
 | |
| #ifndef QDEV_CLOCK_H
 | |
| #define QDEV_CLOCK_H
 | |
| 
 | |
| #include "hw/clock.h"
 | |
| 
 | |
| /**
 | |
|  * qdev_init_clock_in:
 | |
|  * @dev: the device to add an input clock to
 | |
|  * @name: the name of the clock (can't be NULL).
 | |
|  * @callback: optional callback to be called on update or NULL.
 | |
|  * @opaque: argument for the callback
 | |
|  * @returns: a pointer to the newly added clock
 | |
|  *
 | |
|  * Add an input clock to device @dev as a clock named @name.
 | |
|  * This adds a child<> property.
 | |
|  * The callback will be called with @opaque as opaque parameter.
 | |
|  */
 | |
| Clock *qdev_init_clock_in(DeviceState *dev, const char *name,
 | |
|                           ClockCallback *callback, void *opaque);
 | |
| 
 | |
| /**
 | |
|  * qdev_init_clock_out:
 | |
|  * @dev: the device to add an output clock to
 | |
|  * @name: the name of the clock (can't be NULL).
 | |
|  * @returns: a pointer to the newly added clock
 | |
|  *
 | |
|  * Add an output clock to device @dev as a clock named @name.
 | |
|  * This adds a child<> property.
 | |
|  */
 | |
| Clock *qdev_init_clock_out(DeviceState *dev, const char *name);
 | |
| 
 | |
| /**
 | |
|  * qdev_get_clock_in:
 | |
|  * @dev: the device which has the clock
 | |
|  * @name: the name of the clock (can't be NULL).
 | |
|  * @returns: a pointer to the clock
 | |
|  *
 | |
|  * Get the input clock @name from @dev or NULL if does not exist.
 | |
|  */
 | |
| Clock *qdev_get_clock_in(DeviceState *dev, const char *name);
 | |
| 
 | |
| /**
 | |
|  * qdev_get_clock_out:
 | |
|  * @dev: the device which has the clock
 | |
|  * @name: the name of the clock (can't be NULL).
 | |
|  * @returns: a pointer to the clock
 | |
|  *
 | |
|  * Get the output clock @name from @dev or NULL if does not exist.
 | |
|  */
 | |
| Clock *qdev_get_clock_out(DeviceState *dev, const char *name);
 | |
| 
 | |
| /**
 | |
|  * qdev_connect_clock_in:
 | |
|  * @dev: a device
 | |
|  * @name: the name of an input clock in @dev
 | |
|  * @source: the source clock (an output clock of another device for example)
 | |
|  *
 | |
|  * Set the source clock of input clock @name of device @dev to @source.
 | |
|  * @source period update will be propagated to @name clock.
 | |
|  *
 | |
|  * Must be called before @dev is realized.
 | |
|  */
 | |
| void qdev_connect_clock_in(DeviceState *dev, const char *name, Clock *source);
 | |
| 
 | |
| /**
 | |
|  * qdev_alias_clock:
 | |
|  * @dev: the device which has the clock
 | |
|  * @name: the name of the clock in @dev (can't be NULL)
 | |
|  * @alias_dev: the device to add the clock
 | |
|  * @alias_name: the name of the clock in @container
 | |
|  * @returns: a pointer to the clock
 | |
|  *
 | |
|  * Add a clock @alias_name in @alias_dev which is an alias of the clock @name
 | |
|  * in @dev. The direction _in_ or _out_ will the same as the original.
 | |
|  * An alias clock must not be modified or used by @alias_dev and should
 | |
|  * typically be only only for device composition purpose.
 | |
|  */
 | |
| Clock *qdev_alias_clock(DeviceState *dev, const char *name,
 | |
|                         DeviceState *alias_dev, const char *alias_name);
 | |
| 
 | |
| /**
 | |
|  * qdev_finalize_clocklist:
 | |
|  * @dev: the device being finalized
 | |
|  *
 | |
|  * Clear the clocklist from @dev. Only used internally in qdev.
 | |
|  */
 | |
| void qdev_finalize_clocklist(DeviceState *dev);
 | |
| 
 | |
| /**
 | |
|  * ClockPortInitElem:
 | |
|  * @name: name of the clock (can't be NULL)
 | |
|  * @output: indicates whether the clock is input or output
 | |
|  * @callback: for inputs, optional callback to be called on clock's update
 | |
|  * with device as opaque
 | |
|  * @offset: optional offset to store the ClockIn or ClockOut pointer in device
 | |
|  * state structure (0 means unused)
 | |
|  */
 | |
| struct ClockPortInitElem {
 | |
|     const char *name;
 | |
|     bool is_output;
 | |
|     ClockCallback *callback;
 | |
|     size_t offset;
 | |
| };
 | |
| 
 | |
| #define clock_offset_value(devstate, field) \
 | |
|     (offsetof(devstate, field) + \
 | |
|      type_check(Clock *, typeof_field(devstate, field)))
 | |
| 
 | |
| #define QDEV_CLOCK(out_not_in, devstate, field, cb) { \
 | |
|     .name = (stringify(field)), \
 | |
|     .is_output = out_not_in, \
 | |
|     .callback = cb, \
 | |
|     .offset = clock_offset_value(devstate, field), \
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * QDEV_CLOCK_(IN|OUT):
 | |
|  * @devstate: structure type. @dev argument of qdev_init_clocks below must be
 | |
|  * a pointer to that same type.
 | |
|  * @field: a field in @_devstate (must be Clock*)
 | |
|  * @callback: (for input only) callback (or NULL) to be called with the device
 | |
|  * state as argument
 | |
|  *
 | |
|  * The name of the clock will be derived from @field
 | |
|  */
 | |
| #define QDEV_CLOCK_IN(devstate, field, callback) \
 | |
|     QDEV_CLOCK(false, devstate, field, callback)
 | |
| 
 | |
| #define QDEV_CLOCK_OUT(devstate, field) \
 | |
|     QDEV_CLOCK(true, devstate, field, NULL)
 | |
| 
 | |
| #define QDEV_CLOCK_END { .name = NULL }
 | |
| 
 | |
| typedef struct ClockPortInitElem ClockPortInitArray[];
 | |
| 
 | |
| /**
 | |
|  * qdev_init_clocks:
 | |
|  * @dev: the device to add clocks to
 | |
|  * @clocks: a QDEV_CLOCK_END-terminated array which contains the
 | |
|  * clocks information.
 | |
|  */
 | |
| void qdev_init_clocks(DeviceState *dev, const ClockPortInitArray clocks);
 | |
| 
 | |
| #endif /* QDEV_CLOCK_H */
 |