Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

drm: add drmm_encoder_alloc()

Add an alternative to drm_encoder_init() that allocates and initializes
an encoder and registers drm_encoder_cleanup() with
drmm_add_action_or_reset().

Signed-off-by: Philipp Zabel <p.zabel@pengutronix.de>
Reviewed-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
Reviewed-by: Daniel Vetter <daniel.vetter@ffwll.ch>

+116 -23
+86 -23
drivers/gpu/drm/drm_encoder.c
··· 26 26 #include <drm/drm_device.h> 27 27 #include <drm/drm_drv.h> 28 28 #include <drm/drm_encoder.h> 29 + #include <drm/drm_managed.h> 29 30 30 31 #include "drm_crtc_internal.h" 31 32 ··· 92 91 } 93 92 } 94 93 95 - /** 96 - * drm_encoder_init - Init a preallocated encoder 97 - * @dev: drm device 98 - * @encoder: the encoder to init 99 - * @funcs: callbacks for this encoder 100 - * @encoder_type: user visible type of the encoder 101 - * @name: printf style format string for the encoder name, or NULL for default name 102 - * 103 - * Initialises a preallocated encoder. Encoder should be subclassed as part of 104 - * driver encoder objects. At driver unload time drm_encoder_cleanup() should be 105 - * called from the driver's &drm_encoder_funcs.destroy hook. 106 - * 107 - * Returns: 108 - * Zero on success, error code on failure. 109 - */ 110 - int drm_encoder_init(struct drm_device *dev, 111 - struct drm_encoder *encoder, 112 - const struct drm_encoder_funcs *funcs, 113 - int encoder_type, const char *name, ...) 94 + __printf(5, 0) 95 + static int __drm_encoder_init(struct drm_device *dev, 96 + struct drm_encoder *encoder, 97 + const struct drm_encoder_funcs *funcs, 98 + int encoder_type, const char *name, va_list ap) 114 99 { 115 100 int ret; 116 101 ··· 112 125 encoder->encoder_type = encoder_type; 113 126 encoder->funcs = funcs; 114 127 if (name) { 115 - va_list ap; 116 - 117 - va_start(ap, name); 118 128 encoder->name = kvasprintf(GFP_KERNEL, name, ap); 119 - va_end(ap); 120 129 } else { 121 130 encoder->name = kasprintf(GFP_KERNEL, "%s-%d", 122 131 drm_encoder_enum_list[encoder_type].name, ··· 130 147 out_put: 131 148 if (ret) 132 149 drm_mode_object_unregister(dev, &encoder->base); 150 + 151 + return ret; 152 + } 153 + 154 + /** 155 + * drm_encoder_init - Init a preallocated encoder 156 + * @dev: drm device 157 + * @encoder: the encoder to init 158 + * @funcs: callbacks for this encoder 159 + * @encoder_type: user visible type of the encoder 160 + * @name: printf style format string for the encoder name, or NULL for default name 161 + * 162 + * Initializes a preallocated encoder. Encoder should be subclassed as part of 163 + * driver encoder objects. At driver unload time the driver's 164 + * &drm_encoder_funcs.destroy hook should call drm_encoder_cleanup() and kfree() 165 + * the encoder structure. The encoder structure should not be allocated with 166 + * devm_kzalloc(). 167 + * 168 + * Note: consider using drmm_encoder_alloc() instead of drm_encoder_init() to 169 + * let the DRM managed resource infrastructure take care of cleanup and 170 + * deallocation. 171 + * 172 + * Returns: 173 + * Zero on success, error code on failure. 174 + */ 175 + int drm_encoder_init(struct drm_device *dev, 176 + struct drm_encoder *encoder, 177 + const struct drm_encoder_funcs *funcs, 178 + int encoder_type, const char *name, ...) 179 + { 180 + va_list ap; 181 + int ret; 182 + 183 + WARN_ON(!funcs->destroy); 184 + 185 + va_start(ap, name); 186 + ret = __drm_encoder_init(dev, encoder, funcs, encoder_type, name, ap); 187 + va_end(ap); 133 188 134 189 return ret; 135 190 } ··· 201 180 memset(encoder, 0, sizeof(*encoder)); 202 181 } 203 182 EXPORT_SYMBOL(drm_encoder_cleanup); 183 + 184 + static void drmm_encoder_alloc_release(struct drm_device *dev, void *ptr) 185 + { 186 + struct drm_encoder *encoder = ptr; 187 + 188 + if (WARN_ON(!encoder->dev)) 189 + return; 190 + 191 + drm_encoder_cleanup(encoder); 192 + } 193 + 194 + void *__drmm_encoder_alloc(struct drm_device *dev, size_t size, size_t offset, 195 + const struct drm_encoder_funcs *funcs, 196 + int encoder_type, const char *name, ...) 197 + { 198 + void *container; 199 + struct drm_encoder *encoder; 200 + va_list ap; 201 + int ret; 202 + 203 + if (WARN_ON(funcs && funcs->destroy)) 204 + return ERR_PTR(-EINVAL); 205 + 206 + container = drmm_kzalloc(dev, size, GFP_KERNEL); 207 + if (!container) 208 + return ERR_PTR(-EINVAL); 209 + 210 + encoder = container + offset; 211 + 212 + va_start(ap, name); 213 + ret = __drm_encoder_init(dev, encoder, funcs, encoder_type, name, ap); 214 + va_end(ap); 215 + if (ret) 216 + return ERR_PTR(ret); 217 + 218 + ret = drmm_add_action_or_reset(dev, drmm_encoder_alloc_release, encoder); 219 + if (ret) 220 + return ERR_PTR(ret); 221 + 222 + return container; 223 + } 224 + EXPORT_SYMBOL(__drmm_encoder_alloc); 204 225 205 226 static struct drm_crtc *drm_encoder_get_crtc(struct drm_encoder *encoder) 206 227 {
+30
include/drm/drm_encoder.h
··· 194 194 const struct drm_encoder_funcs *funcs, 195 195 int encoder_type, const char *name, ...); 196 196 197 + __printf(6, 7) 198 + void *__drmm_encoder_alloc(struct drm_device *dev, 199 + size_t size, size_t offset, 200 + const struct drm_encoder_funcs *funcs, 201 + int encoder_type, 202 + const char *name, ...); 203 + 204 + /** 205 + * drmm_encoder_alloc - Allocate and initialize an encoder 206 + * @dev: drm device 207 + * @type: the type of the struct which contains struct &drm_encoder 208 + * @member: the name of the &drm_encoder within @type 209 + * @funcs: callbacks for this encoder (optional) 210 + * @encoder_type: user visible type of the encoder 211 + * @name: printf style format string for the encoder name, or NULL for default name 212 + * 213 + * Allocates and initializes an encoder. Encoder should be subclassed as part of 214 + * driver encoder objects. Cleanup is automatically handled through registering 215 + * drm_encoder_cleanup() with drmm_add_action(). 216 + * 217 + * The @drm_encoder_funcs.destroy hook must be NULL. 218 + * 219 + * Returns: 220 + * Pointer to new encoder, or ERR_PTR on failure. 221 + */ 222 + #define drmm_encoder_alloc(dev, type, member, funcs, encoder_type, name, ...) \ 223 + ((type *)__drmm_encoder_alloc(dev, sizeof(type), \ 224 + offsetof(type, member), funcs, \ 225 + encoder_type, name, ##__VA_ARGS__)) 226 + 197 227 /** 198 228 * drm_encoder_index - find the index of a registered encoder 199 229 * @encoder: encoder to find index for