1*bbecb9d1SAndroid Build Coastguard Worker /*
2*bbecb9d1SAndroid Build Coastguard Worker * Copyright 2022 Google LLC
3*bbecb9d1SAndroid Build Coastguard Worker * SPDX-License-Identifier: MIT
4*bbecb9d1SAndroid Build Coastguard Worker */
5*bbecb9d1SAndroid Build Coastguard Worker
6*bbecb9d1SAndroid Build Coastguard Worker #include "config.h"
7*bbecb9d1SAndroid Build Coastguard Worker
8*bbecb9d1SAndroid Build Coastguard Worker #include <errno.h>
9*bbecb9d1SAndroid Build Coastguard Worker #include <inttypes.h>
10*bbecb9d1SAndroid Build Coastguard Worker #include <stddef.h>
11*bbecb9d1SAndroid Build Coastguard Worker #include <stdint.h>
12*bbecb9d1SAndroid Build Coastguard Worker #include <unistd.h>
13*bbecb9d1SAndroid Build Coastguard Worker
14*bbecb9d1SAndroid Build Coastguard Worker #include <xf86drm.h>
15*bbecb9d1SAndroid Build Coastguard Worker
16*bbecb9d1SAndroid Build Coastguard Worker #include "drm_hw.h"
17*bbecb9d1SAndroid Build Coastguard Worker #include "drm_renderer.h"
18*bbecb9d1SAndroid Build Coastguard Worker
19*bbecb9d1SAndroid Build Coastguard Worker #ifdef ENABLE_DRM_MSM
20*bbecb9d1SAndroid Build Coastguard Worker # include "msm/msm_renderer.h"
21*bbecb9d1SAndroid Build Coastguard Worker #endif
22*bbecb9d1SAndroid Build Coastguard Worker
23*bbecb9d1SAndroid Build Coastguard Worker static struct virgl_renderer_capset_drm capset;
24*bbecb9d1SAndroid Build Coastguard Worker
25*bbecb9d1SAndroid Build Coastguard Worker static const struct backend {
26*bbecb9d1SAndroid Build Coastguard Worker uint32_t context_type;
27*bbecb9d1SAndroid Build Coastguard Worker const char *name;
28*bbecb9d1SAndroid Build Coastguard Worker int (*probe)(int fd, struct virgl_renderer_capset_drm *capset);
29*bbecb9d1SAndroid Build Coastguard Worker struct virgl_context *(*create)(int fd);
30*bbecb9d1SAndroid Build Coastguard Worker } backends[] = {
31*bbecb9d1SAndroid Build Coastguard Worker #ifdef ENABLE_DRM_MSM
32*bbecb9d1SAndroid Build Coastguard Worker {
33*bbecb9d1SAndroid Build Coastguard Worker .context_type = VIRTGPU_DRM_CONTEXT_MSM,
34*bbecb9d1SAndroid Build Coastguard Worker .name = "msm",
35*bbecb9d1SAndroid Build Coastguard Worker .probe = msm_renderer_probe,
36*bbecb9d1SAndroid Build Coastguard Worker .create = msm_renderer_create,
37*bbecb9d1SAndroid Build Coastguard Worker },
38*bbecb9d1SAndroid Build Coastguard Worker #endif
39*bbecb9d1SAndroid Build Coastguard Worker };
40*bbecb9d1SAndroid Build Coastguard Worker
41*bbecb9d1SAndroid Build Coastguard Worker int
drm_renderer_init(int drm_fd)42*bbecb9d1SAndroid Build Coastguard Worker drm_renderer_init(int drm_fd)
43*bbecb9d1SAndroid Build Coastguard Worker {
44*bbecb9d1SAndroid Build Coastguard Worker for (unsigned i = 0; i < ARRAY_SIZE(backends); i++) {
45*bbecb9d1SAndroid Build Coastguard Worker const struct backend *b = &backends[i];
46*bbecb9d1SAndroid Build Coastguard Worker int fd;
47*bbecb9d1SAndroid Build Coastguard Worker
48*bbecb9d1SAndroid Build Coastguard Worker if (drm_fd != -1) {
49*bbecb9d1SAndroid Build Coastguard Worker fd = drm_fd;
50*bbecb9d1SAndroid Build Coastguard Worker } else {
51*bbecb9d1SAndroid Build Coastguard Worker fd = drmOpenWithType(b->name, NULL, DRM_NODE_RENDER);
52*bbecb9d1SAndroid Build Coastguard Worker if (fd < 0)
53*bbecb9d1SAndroid Build Coastguard Worker continue;
54*bbecb9d1SAndroid Build Coastguard Worker }
55*bbecb9d1SAndroid Build Coastguard Worker
56*bbecb9d1SAndroid Build Coastguard Worker drmVersionPtr ver = drmGetVersion(fd);
57*bbecb9d1SAndroid Build Coastguard Worker if (!ver) {
58*bbecb9d1SAndroid Build Coastguard Worker close(fd);
59*bbecb9d1SAndroid Build Coastguard Worker return -ENOMEM;
60*bbecb9d1SAndroid Build Coastguard Worker }
61*bbecb9d1SAndroid Build Coastguard Worker
62*bbecb9d1SAndroid Build Coastguard Worker if (strcmp(ver->name, b->name)) {
63*bbecb9d1SAndroid Build Coastguard Worker /* In the drmOpenWithType() path, we will only get back an fd
64*bbecb9d1SAndroid Build Coastguard Worker * for the device with matching name. But when we are using
65*bbecb9d1SAndroid Build Coastguard Worker * an externally provided fd, we need to go thru the backends
66*bbecb9d1SAndroid Build Coastguard Worker * table to see which one has the matching name.
67*bbecb9d1SAndroid Build Coastguard Worker */
68*bbecb9d1SAndroid Build Coastguard Worker assert(drm_fd != -1);
69*bbecb9d1SAndroid Build Coastguard Worker drmFreeVersion(ver);
70*bbecb9d1SAndroid Build Coastguard Worker continue;
71*bbecb9d1SAndroid Build Coastguard Worker }
72*bbecb9d1SAndroid Build Coastguard Worker
73*bbecb9d1SAndroid Build Coastguard Worker capset.version_major = ver->version_major;
74*bbecb9d1SAndroid Build Coastguard Worker capset.version_minor = ver->version_minor;
75*bbecb9d1SAndroid Build Coastguard Worker capset.version_patchlevel = ver->version_patchlevel;
76*bbecb9d1SAndroid Build Coastguard Worker capset.context_type = b->context_type;
77*bbecb9d1SAndroid Build Coastguard Worker
78*bbecb9d1SAndroid Build Coastguard Worker int ret = b->probe(fd, &capset);
79*bbecb9d1SAndroid Build Coastguard Worker if (ret)
80*bbecb9d1SAndroid Build Coastguard Worker memset(&capset, 0, sizeof(capset));
81*bbecb9d1SAndroid Build Coastguard Worker
82*bbecb9d1SAndroid Build Coastguard Worker drmFreeVersion(ver);
83*bbecb9d1SAndroid Build Coastguard Worker close(fd);
84*bbecb9d1SAndroid Build Coastguard Worker return ret;
85*bbecb9d1SAndroid Build Coastguard Worker }
86*bbecb9d1SAndroid Build Coastguard Worker
87*bbecb9d1SAndroid Build Coastguard Worker if (drm_fd != -1)
88*bbecb9d1SAndroid Build Coastguard Worker close(drm_fd);
89*bbecb9d1SAndroid Build Coastguard Worker
90*bbecb9d1SAndroid Build Coastguard Worker return -ENODEV;
91*bbecb9d1SAndroid Build Coastguard Worker }
92*bbecb9d1SAndroid Build Coastguard Worker
93*bbecb9d1SAndroid Build Coastguard Worker void
drm_renderer_fini(void)94*bbecb9d1SAndroid Build Coastguard Worker drm_renderer_fini(void)
95*bbecb9d1SAndroid Build Coastguard Worker {
96*bbecb9d1SAndroid Build Coastguard Worker drm_log("");
97*bbecb9d1SAndroid Build Coastguard Worker }
98*bbecb9d1SAndroid Build Coastguard Worker
99*bbecb9d1SAndroid Build Coastguard Worker void
drm_renderer_reset(void)100*bbecb9d1SAndroid Build Coastguard Worker drm_renderer_reset(void)
101*bbecb9d1SAndroid Build Coastguard Worker {
102*bbecb9d1SAndroid Build Coastguard Worker drm_log("");
103*bbecb9d1SAndroid Build Coastguard Worker }
104*bbecb9d1SAndroid Build Coastguard Worker
105*bbecb9d1SAndroid Build Coastguard Worker size_t
drm_renderer_capset(void * _c)106*bbecb9d1SAndroid Build Coastguard Worker drm_renderer_capset(void *_c)
107*bbecb9d1SAndroid Build Coastguard Worker {
108*bbecb9d1SAndroid Build Coastguard Worker struct virgl_renderer_capset_drm *c = _c;
109*bbecb9d1SAndroid Build Coastguard Worker drm_log("c=%p", c);
110*bbecb9d1SAndroid Build Coastguard Worker
111*bbecb9d1SAndroid Build Coastguard Worker if (!capset.context_type)
112*bbecb9d1SAndroid Build Coastguard Worker return 0;
113*bbecb9d1SAndroid Build Coastguard Worker
114*bbecb9d1SAndroid Build Coastguard Worker if (c)
115*bbecb9d1SAndroid Build Coastguard Worker *c = capset;
116*bbecb9d1SAndroid Build Coastguard Worker
117*bbecb9d1SAndroid Build Coastguard Worker return sizeof(*c);
118*bbecb9d1SAndroid Build Coastguard Worker }
119*bbecb9d1SAndroid Build Coastguard Worker
120*bbecb9d1SAndroid Build Coastguard Worker struct virgl_context *
drm_renderer_create(UNUSED size_t debug_len,UNUSED const char * debug_name)121*bbecb9d1SAndroid Build Coastguard Worker drm_renderer_create(UNUSED size_t debug_len, UNUSED const char *debug_name)
122*bbecb9d1SAndroid Build Coastguard Worker {
123*bbecb9d1SAndroid Build Coastguard Worker for (unsigned i = 0; i < ARRAY_SIZE(backends); i++) {
124*bbecb9d1SAndroid Build Coastguard Worker const struct backend *b = &backends[i];
125*bbecb9d1SAndroid Build Coastguard Worker
126*bbecb9d1SAndroid Build Coastguard Worker if (b->context_type != capset.context_type)
127*bbecb9d1SAndroid Build Coastguard Worker continue;
128*bbecb9d1SAndroid Build Coastguard Worker
129*bbecb9d1SAndroid Build Coastguard Worker int fd = drmOpenWithType(b->name, NULL, DRM_NODE_RENDER);
130*bbecb9d1SAndroid Build Coastguard Worker if (fd < 0)
131*bbecb9d1SAndroid Build Coastguard Worker return NULL;
132*bbecb9d1SAndroid Build Coastguard Worker
133*bbecb9d1SAndroid Build Coastguard Worker return b->create(fd);
134*bbecb9d1SAndroid Build Coastguard Worker }
135*bbecb9d1SAndroid Build Coastguard Worker
136*bbecb9d1SAndroid Build Coastguard Worker return NULL;
137*bbecb9d1SAndroid Build Coastguard Worker }
138