xref: /aosp_15_r20/external/virglrenderer/src/drm/drm_renderer.c (revision bbecb9d118dfdb95f99bd754f8fa9be01f189df3)
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