mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	drm/nouveau/disp: add connector class
Will be used to provide more solid driver interfaces in general, but the immediate motivation is work towards fixing issues with handling hotplug/DP IRQ events. Its use is currently limited to where we support non-polled hotplug already (ie. any GPU since NV40ish era, where our DCB handling works well enough), until that gets cleaned up someday. v2: - use ?: (lyude) Signed-off-by: Ben Skeggs <bskeggs@redhat.com> Reviewed-by: Lyude Paul <lyude@redhat.com> Signed-off-by: Dave Airlie <airlied@redhat.com>
This commit is contained in:
		
							parent
							
								
									889fcbe949
								
							
						
					
					
						commit
						95983aea80
					
				| @ -33,6 +33,7 @@ | |||||||
| #define NVIF_CLASS_VMM_GP100                         /* ifc00d.h */  0x8000c00d | #define NVIF_CLASS_VMM_GP100                         /* ifc00d.h */  0x8000c00d | ||||||
| 
 | 
 | ||||||
| #define NVIF_CLASS_DISP                              /* if0010.h */  0x80000010 | #define NVIF_CLASS_DISP                              /* if0010.h */  0x80000010 | ||||||
|  | #define NVIF_CLASS_CONN                              /* if0011.h */  0x80000011 | ||||||
| #define NVIF_CLASS_DISP_CHAN                         /* if0014.h */  0x80000014 | #define NVIF_CLASS_DISP_CHAN                         /* if0014.h */  0x80000014 | ||||||
| 
 | 
 | ||||||
| /* the below match nvidia-assigned (either in hw, or sw) class numbers */ | /* the below match nvidia-assigned (either in hw, or sw) class numbers */ | ||||||
|  | |||||||
							
								
								
									
										13
									
								
								drivers/gpu/drm/nouveau/include/nvif/conn.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								drivers/gpu/drm/nouveau/include/nvif/conn.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,13 @@ | |||||||
|  | /* SPDX-License-Identifier: MIT */ | ||||||
|  | #ifndef __NVIF_CONN_H__ | ||||||
|  | #define __NVIF_CONN_H__ | ||||||
|  | #include <nvif/object.h> | ||||||
|  | struct nvif_disp; | ||||||
|  | 
 | ||||||
|  | struct nvif_conn { | ||||||
|  | 	struct nvif_object object; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | int nvif_conn_ctor(struct nvif_disp *, const char *name, int id, struct nvif_conn *); | ||||||
|  | void nvif_conn_dtor(struct nvif_conn *); | ||||||
|  | #endif | ||||||
| @ -5,6 +5,7 @@ struct nvif_device; | |||||||
| 
 | 
 | ||||||
| struct nvif_disp { | struct nvif_disp { | ||||||
| 	struct nvif_object object; | 	struct nvif_object object; | ||||||
|  | 	unsigned long conn_mask; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| int nvif_disp_ctor(struct nvif_device *, const char *name, s32 oclass, | int nvif_disp_ctor(struct nvif_device *, const char *name, s32 oclass, | ||||||
|  | |||||||
| @ -5,7 +5,8 @@ | |||||||
| union nvif_disp_args { | union nvif_disp_args { | ||||||
| 	struct nvif_disp_v0 { | 	struct nvif_disp_v0 { | ||||||
| 		__u8 version; | 		__u8 version; | ||||||
| 		__u8 pad01[7]; | 		__u8 pad01[3]; | ||||||
|  | 		__u32 conn_mask; | ||||||
| 	} v0; | 	} v0; | ||||||
| }; | }; | ||||||
| #endif | #endif | ||||||
|  | |||||||
							
								
								
									
										12
									
								
								drivers/gpu/drm/nouveau/include/nvif/if0011.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								drivers/gpu/drm/nouveau/include/nvif/if0011.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,12 @@ | |||||||
|  | /* SPDX-License-Identifier: MIT */ | ||||||
|  | #ifndef __NVIF_IF0011_H__ | ||||||
|  | #define __NVIF_IF0011_H__ | ||||||
|  | 
 | ||||||
|  | union nvif_conn_args { | ||||||
|  | 	struct nvif_conn_v0 { | ||||||
|  | 		__u8 version; | ||||||
|  | 		__u8 id;	/* DCB connector table index. */ | ||||||
|  | 		__u8 pad02[6]; | ||||||
|  | 	} v0; | ||||||
|  | }; | ||||||
|  | #endif | ||||||
| @ -404,6 +404,7 @@ nouveau_connector_destroy(struct drm_connector *connector) | |||||||
| 		drm_dp_cec_unregister_connector(&nv_connector->aux); | 		drm_dp_cec_unregister_connector(&nv_connector->aux); | ||||||
| 		kfree(nv_connector->aux.name); | 		kfree(nv_connector->aux.name); | ||||||
| 	} | 	} | ||||||
|  | 	nvif_conn_dtor(&nv_connector->conn); | ||||||
| 	kfree(connector); | 	kfree(connector); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -1388,6 +1389,15 @@ nouveau_connector_create(struct drm_device *dev, | |||||||
| 	drm_connector_init(dev, connector, funcs, type); | 	drm_connector_init(dev, connector, funcs, type); | ||||||
| 	drm_connector_helper_add(connector, &nouveau_connector_helper_funcs); | 	drm_connector_helper_add(connector, &nouveau_connector_helper_funcs); | ||||||
| 
 | 
 | ||||||
|  | 	if (nv_connector->dcb && (disp->disp.conn_mask & BIT(nv_connector->index))) { | ||||||
|  | 		ret = nvif_conn_ctor(&disp->disp, nv_connector->base.name, nv_connector->index, | ||||||
|  | 				     &nv_connector->conn); | ||||||
|  | 		if (ret) { | ||||||
|  | 			kfree(nv_connector); | ||||||
|  | 			return ERR_PTR(ret); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	connector->funcs->reset(connector); | 	connector->funcs->reset(connector); | ||||||
| 	nouveau_conn_attach_properties(connector); | 	nouveau_conn_attach_properties(connector); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -26,7 +26,7 @@ | |||||||
| 
 | 
 | ||||||
| #ifndef __NOUVEAU_CONNECTOR_H__ | #ifndef __NOUVEAU_CONNECTOR_H__ | ||||||
| #define __NOUVEAU_CONNECTOR_H__ | #define __NOUVEAU_CONNECTOR_H__ | ||||||
| 
 | #include <nvif/conn.h> | ||||||
| #include <nvif/notify.h> | #include <nvif/notify.h> | ||||||
| 
 | 
 | ||||||
| #include <nvhw/class/cl507d.h> | #include <nvhw/class/cl507d.h> | ||||||
| @ -123,6 +123,7 @@ struct nouveau_connector { | |||||||
| 	u8 index; | 	u8 index; | ||||||
| 	u8 *dcb; | 	u8 *dcb; | ||||||
| 
 | 
 | ||||||
|  | 	struct nvif_conn conn; | ||||||
| 	struct nvif_notify hpd; | 	struct nvif_notify hpd; | ||||||
| 
 | 
 | ||||||
| 	struct drm_dp_aux aux; | 	struct drm_dp_aux aux; | ||||||
|  | |||||||
| @ -1,6 +1,7 @@ | |||||||
| # SPDX-License-Identifier: MIT | # SPDX-License-Identifier: MIT | ||||||
| nvif-y := nvif/object.o | nvif-y := nvif/object.o | ||||||
| nvif-y += nvif/client.o | nvif-y += nvif/client.o | ||||||
|  | nvif-y += nvif/conn.o | ||||||
| nvif-y += nvif/device.o | nvif-y += nvif/device.o | ||||||
| nvif-y += nvif/disp.o | nvif-y += nvif/disp.o | ||||||
| nvif-y += nvif/driver.o | nvif-y += nvif/driver.o | ||||||
|  | |||||||
							
								
								
									
										48
									
								
								drivers/gpu/drm/nouveau/nvif/conn.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								drivers/gpu/drm/nouveau/nvif/conn.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,48 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright 2021 Red Hat Inc. | ||||||
|  |  * | ||||||
|  |  * 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 COPYRIGHT HOLDER(S) OR AUTHOR(S) 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. | ||||||
|  |  */ | ||||||
|  | #include <nvif/conn.h> | ||||||
|  | #include <nvif/disp.h> | ||||||
|  | #include <nvif/printf.h> | ||||||
|  | 
 | ||||||
|  | #include <nvif/class.h> | ||||||
|  | #include <nvif/if0011.h> | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | nvif_conn_dtor(struct nvif_conn *conn) | ||||||
|  | { | ||||||
|  | 	nvif_object_dtor(&conn->object); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | nvif_conn_ctor(struct nvif_disp *disp, const char *name, int id, struct nvif_conn *conn) | ||||||
|  | { | ||||||
|  | 	struct nvif_conn_v0 args; | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 	args.version = 0; | ||||||
|  | 	args.id = id; | ||||||
|  | 
 | ||||||
|  | 	ret = nvif_object_ctor(&disp->object, name ?: "nvifConn", id, NVIF_CLASS_CONN, | ||||||
|  | 			       &args, sizeof(args), &conn->object); | ||||||
|  | 	NVIF_ERRON(ret, &disp->object, "[NEW conn id:%d]", id); | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
| @ -69,5 +69,10 @@ nvif_disp_ctor(struct nvif_device *device, const char *name, s32 oclass, struct | |||||||
| 	ret = nvif_object_ctor(&device->object, name ?: "nvifDisp", 0, | 	ret = nvif_object_ctor(&device->object, name ?: "nvifDisp", 0, | ||||||
| 			       disps[cid].oclass, &args, sizeof(args), &disp->object); | 			       disps[cid].oclass, &args, sizeof(args), &disp->object); | ||||||
| 	NVIF_ERRON(ret, &device->object, "[NEW disp%04x]", disps[cid].oclass); | 	NVIF_ERRON(ret, &device->object, "[NEW disp%04x]", disps[cid].oclass); | ||||||
|  | 	if (ret) | ||||||
| 		return ret; | 		return ret; | ||||||
|  | 
 | ||||||
|  | 	NVIF_DEBUG(&disp->object, "[NEW] conn_mask:%08x", args.conn_mask); | ||||||
|  | 	disp->conn_mask = args.conn_mask; | ||||||
|  | 	return 0; | ||||||
| } | } | ||||||
|  | |||||||
| @ -32,3 +32,4 @@ nvkm-y += nvkm/engine/disp/rootnv04.o | |||||||
| nvkm-y += nvkm/engine/disp/rootnv50.o | nvkm-y += nvkm/engine/disp/rootnv50.o | ||||||
| 
 | 
 | ||||||
| nvkm-y += nvkm/engine/disp/udisp.o | nvkm-y += nvkm/engine/disp/udisp.o | ||||||
|  | nvkm-y += nvkm/engine/disp/uconn.o | ||||||
|  | |||||||
| @ -1,7 +1,7 @@ | |||||||
| /* SPDX-License-Identifier: MIT */ | /* SPDX-License-Identifier: MIT */ | ||||||
| #ifndef __NVKM_DISP_CONN_H__ | #ifndef __NVKM_DISP_CONN_H__ | ||||||
| #define __NVKM_DISP_CONN_H__ | #define __NVKM_DISP_CONN_H__ | ||||||
| #include <engine/disp.h> | #include "priv.h" | ||||||
| 
 | 
 | ||||||
| #include <core/notify.h> | #include <core/notify.h> | ||||||
| #include <subdev/bios.h> | #include <subdev/bios.h> | ||||||
| @ -15,6 +15,8 @@ struct nvkm_conn { | |||||||
| 	struct nvkm_notify hpd; | 	struct nvkm_notify hpd; | ||||||
| 
 | 
 | ||||||
| 	struct list_head head; | 	struct list_head head; | ||||||
|  | 
 | ||||||
|  | 	struct nvkm_object object; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| int nvkm_conn_new(struct nvkm_disp *, int index, struct nvbios_connE *, | int nvkm_conn_new(struct nvkm_disp *, int index, struct nvbios_connE *, | ||||||
|  | |||||||
| @ -84,4 +84,5 @@ extern const struct nvkm_event_func gf119_disp_chan_uevent; | |||||||
| extern const struct nvkm_event_func gv100_disp_chan_uevent; | extern const struct nvkm_event_func gv100_disp_chan_uevent; | ||||||
| 
 | 
 | ||||||
| int nvkm_udisp_new(const struct nvkm_oclass *, void *, u32, struct nvkm_object **); | int nvkm_udisp_new(const struct nvkm_oclass *, void *, u32, struct nvkm_object **); | ||||||
|  | int nvkm_uconn_new(const struct nvkm_oclass *, void *, u32, struct nvkm_object **); | ||||||
| #endif | #endif | ||||||
|  | |||||||
							
								
								
									
										74
									
								
								drivers/gpu/drm/nouveau/nvkm/engine/disp/uconn.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								drivers/gpu/drm/nouveau/nvkm/engine/disp/uconn.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,74 @@ | |||||||
|  | /*
 | ||||||
|  |  * Copyright 2021 Red Hat Inc. | ||||||
|  |  * | ||||||
|  |  * 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 COPYRIGHT HOLDER(S) OR AUTHOR(S) 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. | ||||||
|  |  */ | ||||||
|  | #define nvkm_uconn(p) container_of((p), struct nvkm_conn, object) | ||||||
|  | #include "conn.h" | ||||||
|  | 
 | ||||||
|  | #include <nvif/if0011.h> | ||||||
|  | 
 | ||||||
|  | static void * | ||||||
|  | nvkm_uconn_dtor(struct nvkm_object *object) | ||||||
|  | { | ||||||
|  | 	struct nvkm_conn *conn = nvkm_uconn(object); | ||||||
|  | 	struct nvkm_disp *disp = conn->disp; | ||||||
|  | 
 | ||||||
|  | 	spin_lock(&disp->client.lock); | ||||||
|  | 	conn->object.func = NULL; | ||||||
|  | 	spin_unlock(&disp->client.lock); | ||||||
|  | 	return NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static const struct nvkm_object_func | ||||||
|  | nvkm_uconn = { | ||||||
|  | 	.dtor = nvkm_uconn_dtor, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | nvkm_uconn_new(const struct nvkm_oclass *oclass, void *argv, u32 argc, struct nvkm_object **pobject) | ||||||
|  | { | ||||||
|  | 	struct nvkm_disp *disp = nvkm_udisp(oclass->parent); | ||||||
|  | 	struct nvkm_conn *cont, *conn = NULL; | ||||||
|  | 	union nvif_conn_args *args = argv; | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 	if (argc != sizeof(args->v0) || args->v0.version != 0) | ||||||
|  | 		return -ENOSYS; | ||||||
|  | 
 | ||||||
|  | 	list_for_each_entry(cont, &disp->conns, head) { | ||||||
|  | 		if (cont->index == args->v0.id) { | ||||||
|  | 			conn = cont; | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (!conn) | ||||||
|  | 		return -EINVAL; | ||||||
|  | 
 | ||||||
|  | 	ret = -EBUSY; | ||||||
|  | 	spin_lock(&disp->client.lock); | ||||||
|  | 	if (!conn->object.func) { | ||||||
|  | 		nvkm_object_ctor(&nvkm_uconn, oclass, &conn->object); | ||||||
|  | 		*pobject = &conn->object; | ||||||
|  | 		ret = 0; | ||||||
|  | 	} | ||||||
|  | 	spin_unlock(&disp->client.lock); | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
| @ -20,7 +20,9 @@ | |||||||
|  * OTHER DEALINGS IN THE SOFTWARE. |  * OTHER DEALINGS IN THE SOFTWARE. | ||||||
|  */ |  */ | ||||||
| #include "priv.h" | #include "priv.h" | ||||||
|  | #include "conn.h" | ||||||
| 
 | 
 | ||||||
|  | #include <nvif/class.h> | ||||||
| #include <nvif/if0010.h> | #include <nvif/if0010.h> | ||||||
| 
 | 
 | ||||||
| static int | static int | ||||||
| @ -28,6 +30,12 @@ nvkm_udisp_sclass(struct nvkm_object *object, int index, struct nvkm_oclass *scl | |||||||
| { | { | ||||||
| 	struct nvkm_disp *disp = nvkm_udisp(object); | 	struct nvkm_disp *disp = nvkm_udisp(object); | ||||||
| 
 | 
 | ||||||
|  | 	if (index-- == 0) { | ||||||
|  | 		sclass->base = (struct nvkm_sclass) { 0, 0, NVIF_CLASS_CONN }; | ||||||
|  | 		sclass->ctor = nvkm_uconn_new; | ||||||
|  | 		return 0; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	if (disp->func->user[index].ctor) { | 	if (disp->func->user[index].ctor) { | ||||||
| 		sclass->base = disp->func->user[index].base; | 		sclass->base = disp->func->user[index].base; | ||||||
| 		sclass->ctor = disp->func->user[index].ctor; | 		sclass->ctor = disp->func->user[index].ctor; | ||||||
| @ -72,6 +80,7 @@ int | |||||||
| nvkm_udisp_new(const struct nvkm_oclass *oclass, void *argv, u32 argc, struct nvkm_object **pobject) | nvkm_udisp_new(const struct nvkm_oclass *oclass, void *argv, u32 argc, struct nvkm_object **pobject) | ||||||
| { | { | ||||||
| 	struct nvkm_disp *disp = nvkm_disp(oclass->engine); | 	struct nvkm_disp *disp = nvkm_disp(oclass->engine); | ||||||
|  | 	struct nvkm_conn *conn; | ||||||
| 	union nvif_disp_args *args = argv; | 	union nvif_disp_args *args = argv; | ||||||
| 
 | 
 | ||||||
| 	if (argc != sizeof(args->v0) || args->v0.version != 0) | 	if (argc != sizeof(args->v0) || args->v0.version != 0) | ||||||
| @ -85,5 +94,10 @@ nvkm_udisp_new(const struct nvkm_oclass *oclass, void *argv, u32 argc, struct nv | |||||||
| 	nvkm_object_ctor(&nvkm_udisp, oclass, &disp->client.object); | 	nvkm_object_ctor(&nvkm_udisp, oclass, &disp->client.object); | ||||||
| 	*pobject = &disp->client.object; | 	*pobject = &disp->client.object; | ||||||
| 	spin_unlock(&disp->client.lock); | 	spin_unlock(&disp->client.lock); | ||||||
|  | 
 | ||||||
|  | 	args->v0.conn_mask = 0; | ||||||
|  | 	list_for_each_entry(conn, &disp->conns, head) | ||||||
|  | 		args->v0.conn_mask |= BIT(conn->index); | ||||||
|  | 
 | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Ben Skeggs
						Ben Skeggs