xref: /aosp_15_r20/external/mesa3d/src/gbm/main/backend.c (revision 6104692788411f58d303aa86923a9ff6ecaded22)
1*61046927SAndroid Build Coastguard Worker /*
2*61046927SAndroid Build Coastguard Worker  * Copyright © 2011 Intel Corporation
3*61046927SAndroid Build Coastguard Worker  * Copyright © 2021 NVIDIA Corporation
4*61046927SAndroid Build Coastguard Worker  *
5*61046927SAndroid Build Coastguard Worker  * Permission is hereby granted, free of charge, to any person obtaining a
6*61046927SAndroid Build Coastguard Worker  * copy of this software and associated documentation files (the "Software"),
7*61046927SAndroid Build Coastguard Worker  * to deal in the Software without restriction, including without limitation
8*61046927SAndroid Build Coastguard Worker  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
9*61046927SAndroid Build Coastguard Worker  * and/or sell copies of the Software, and to permit persons to whom the
10*61046927SAndroid Build Coastguard Worker  * Software is furnished to do so, subject to the following conditions:
11*61046927SAndroid Build Coastguard Worker  *
12*61046927SAndroid Build Coastguard Worker  * The above copyright notice and this permission notice (including the next
13*61046927SAndroid Build Coastguard Worker  * paragraph) shall be included in all copies or substantial portions of the
14*61046927SAndroid Build Coastguard Worker  * Software.
15*61046927SAndroid Build Coastguard Worker  *
16*61046927SAndroid Build Coastguard Worker  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17*61046927SAndroid Build Coastguard Worker  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18*61046927SAndroid Build Coastguard Worker  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19*61046927SAndroid Build Coastguard Worker  * NONINFRINGEMENT.  IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
20*61046927SAndroid Build Coastguard Worker  * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
21*61046927SAndroid Build Coastguard Worker  * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
22*61046927SAndroid Build Coastguard Worker  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
23*61046927SAndroid Build Coastguard Worker  * DEALINGS IN THE SOFTWARE.
24*61046927SAndroid Build Coastguard Worker  *
25*61046927SAndroid Build Coastguard Worker  * Authors:
26*61046927SAndroid Build Coastguard Worker  *    Benjamin Franzke <[email protected]>
27*61046927SAndroid Build Coastguard Worker  *    James Jones <[email protected]>
28*61046927SAndroid Build Coastguard Worker  */
29*61046927SAndroid Build Coastguard Worker 
30*61046927SAndroid Build Coastguard Worker #include <stdio.h>
31*61046927SAndroid Build Coastguard Worker #include <stddef.h>
32*61046927SAndroid Build Coastguard Worker #include <stdlib.h>
33*61046927SAndroid Build Coastguard Worker #include <string.h>
34*61046927SAndroid Build Coastguard Worker #include <limits.h>
35*61046927SAndroid Build Coastguard Worker #include <assert.h>
36*61046927SAndroid Build Coastguard Worker #include <dlfcn.h>
37*61046927SAndroid Build Coastguard Worker #include <xf86drm.h>
38*61046927SAndroid Build Coastguard Worker 
39*61046927SAndroid Build Coastguard Worker #include "loader.h"
40*61046927SAndroid Build Coastguard Worker #include "backend.h"
41*61046927SAndroid Build Coastguard Worker 
42*61046927SAndroid Build Coastguard Worker #define ARRAY_SIZE(a) (sizeof(a)/sizeof((a)[0]))
43*61046927SAndroid Build Coastguard Worker #define VER_MIN(a, b) ((a) < (b) ? (a) : (b))
44*61046927SAndroid Build Coastguard Worker 
45*61046927SAndroid Build Coastguard Worker struct gbm_backend_desc {
46*61046927SAndroid Build Coastguard Worker    const char *name;
47*61046927SAndroid Build Coastguard Worker    const struct gbm_backend *backend;
48*61046927SAndroid Build Coastguard Worker    void *lib;
49*61046927SAndroid Build Coastguard Worker };
50*61046927SAndroid Build Coastguard Worker 
51*61046927SAndroid Build Coastguard Worker #define BACKEND_LIB_SUFFIX "_gbm"
52*61046927SAndroid Build Coastguard Worker static const char *backend_search_path_vars[] = {
53*61046927SAndroid Build Coastguard Worker    "GBM_BACKENDS_PATH",
54*61046927SAndroid Build Coastguard Worker    NULL
55*61046927SAndroid Build Coastguard Worker };
56*61046927SAndroid Build Coastguard Worker 
57*61046927SAndroid Build Coastguard Worker static void
free_backend_desc(const struct gbm_backend_desc * backend_desc)58*61046927SAndroid Build Coastguard Worker free_backend_desc(const struct gbm_backend_desc *backend_desc)
59*61046927SAndroid Build Coastguard Worker {
60*61046927SAndroid Build Coastguard Worker    assert(backend_desc->lib);
61*61046927SAndroid Build Coastguard Worker 
62*61046927SAndroid Build Coastguard Worker    dlclose(backend_desc->lib);
63*61046927SAndroid Build Coastguard Worker    free((void *)backend_desc->name);
64*61046927SAndroid Build Coastguard Worker    free((void *)backend_desc);
65*61046927SAndroid Build Coastguard Worker }
66*61046927SAndroid Build Coastguard Worker 
67*61046927SAndroid Build Coastguard Worker static struct gbm_backend_desc *
create_backend_desc(const char * name,const struct gbm_backend * backend,void * lib)68*61046927SAndroid Build Coastguard Worker create_backend_desc(const char *name,
69*61046927SAndroid Build Coastguard Worker                     const struct gbm_backend *backend,
70*61046927SAndroid Build Coastguard Worker                     void *lib)
71*61046927SAndroid Build Coastguard Worker {
72*61046927SAndroid Build Coastguard Worker    struct gbm_backend_desc *new_desc = calloc(1, sizeof(*new_desc));
73*61046927SAndroid Build Coastguard Worker 
74*61046927SAndroid Build Coastguard Worker    if (!new_desc)
75*61046927SAndroid Build Coastguard Worker       return NULL;
76*61046927SAndroid Build Coastguard Worker 
77*61046927SAndroid Build Coastguard Worker    new_desc->name = strdup(name);
78*61046927SAndroid Build Coastguard Worker 
79*61046927SAndroid Build Coastguard Worker    if (!new_desc->name) {
80*61046927SAndroid Build Coastguard Worker       free(new_desc);
81*61046927SAndroid Build Coastguard Worker       return NULL;
82*61046927SAndroid Build Coastguard Worker    }
83*61046927SAndroid Build Coastguard Worker 
84*61046927SAndroid Build Coastguard Worker    new_desc->backend = backend;
85*61046927SAndroid Build Coastguard Worker    new_desc->lib = lib;
86*61046927SAndroid Build Coastguard Worker 
87*61046927SAndroid Build Coastguard Worker    return new_desc;
88*61046927SAndroid Build Coastguard Worker }
89*61046927SAndroid Build Coastguard Worker 
90*61046927SAndroid Build Coastguard Worker static struct gbm_device *
backend_create_device(const struct gbm_backend_desc * bd,int fd)91*61046927SAndroid Build Coastguard Worker backend_create_device(const struct gbm_backend_desc *bd, int fd)
92*61046927SAndroid Build Coastguard Worker {
93*61046927SAndroid Build Coastguard Worker    const uint32_t abi_ver = VER_MIN(GBM_BACKEND_ABI_VERSION,
94*61046927SAndroid Build Coastguard Worker                                     bd->backend->v0.backend_version);
95*61046927SAndroid Build Coastguard Worker    struct gbm_device *dev = bd->backend->v0.create_device(fd, abi_ver);
96*61046927SAndroid Build Coastguard Worker 
97*61046927SAndroid Build Coastguard Worker    if (dev) {
98*61046927SAndroid Build Coastguard Worker       if (abi_ver != dev->v0.backend_version) {
99*61046927SAndroid Build Coastguard Worker          _gbm_device_destroy(dev);
100*61046927SAndroid Build Coastguard Worker          return NULL;
101*61046927SAndroid Build Coastguard Worker       }
102*61046927SAndroid Build Coastguard Worker       dev->v0.backend_desc = bd;
103*61046927SAndroid Build Coastguard Worker    }
104*61046927SAndroid Build Coastguard Worker 
105*61046927SAndroid Build Coastguard Worker    return dev;
106*61046927SAndroid Build Coastguard Worker }
107*61046927SAndroid Build Coastguard Worker 
108*61046927SAndroid Build Coastguard Worker static struct gbm_device *
load_backend_by_name(const char * name,int fd,bool warn_on_fail)109*61046927SAndroid Build Coastguard Worker load_backend_by_name(const char *name, int fd, bool warn_on_fail)
110*61046927SAndroid Build Coastguard Worker {
111*61046927SAndroid Build Coastguard Worker    void *lib = loader_open_driver_lib(name, BACKEND_LIB_SUFFIX,
112*61046927SAndroid Build Coastguard Worker                                       backend_search_path_vars,
113*61046927SAndroid Build Coastguard Worker                                       DEFAULT_BACKENDS_PATH,
114*61046927SAndroid Build Coastguard Worker                                       warn_on_fail);
115*61046927SAndroid Build Coastguard Worker 
116*61046927SAndroid Build Coastguard Worker    if (!lib)
117*61046927SAndroid Build Coastguard Worker       return NULL;
118*61046927SAndroid Build Coastguard Worker 
119*61046927SAndroid Build Coastguard Worker    struct gbm_device *dev = NULL;
120*61046927SAndroid Build Coastguard Worker    struct gbm_backend_desc *backend_desc;
121*61046927SAndroid Build Coastguard Worker    const struct gbm_backend *gbm_backend;
122*61046927SAndroid Build Coastguard Worker    GBM_GET_BACKEND_PROC_PTR get_backend;
123*61046927SAndroid Build Coastguard Worker 
124*61046927SAndroid Build Coastguard Worker    get_backend = dlsym(lib, GBM_GET_BACKEND_PROC_NAME);
125*61046927SAndroid Build Coastguard Worker 
126*61046927SAndroid Build Coastguard Worker    if (!get_backend)
127*61046927SAndroid Build Coastguard Worker       goto fail;
128*61046927SAndroid Build Coastguard Worker 
129*61046927SAndroid Build Coastguard Worker    gbm_backend = get_backend(&gbm_core);
130*61046927SAndroid Build Coastguard Worker    backend_desc = create_backend_desc(name, gbm_backend, lib);
131*61046927SAndroid Build Coastguard Worker 
132*61046927SAndroid Build Coastguard Worker    if (!backend_desc)
133*61046927SAndroid Build Coastguard Worker       goto fail;
134*61046927SAndroid Build Coastguard Worker 
135*61046927SAndroid Build Coastguard Worker    dev = backend_create_device(backend_desc, fd);
136*61046927SAndroid Build Coastguard Worker 
137*61046927SAndroid Build Coastguard Worker    if (!dev)
138*61046927SAndroid Build Coastguard Worker       free_backend_desc(backend_desc);
139*61046927SAndroid Build Coastguard Worker 
140*61046927SAndroid Build Coastguard Worker    return dev;
141*61046927SAndroid Build Coastguard Worker 
142*61046927SAndroid Build Coastguard Worker fail:
143*61046927SAndroid Build Coastguard Worker    dlclose(lib);
144*61046927SAndroid Build Coastguard Worker    return NULL;
145*61046927SAndroid Build Coastguard Worker }
146*61046927SAndroid Build Coastguard Worker 
147*61046927SAndroid Build Coastguard Worker struct gbm_device *
_gbm_create_device(int fd)148*61046927SAndroid Build Coastguard Worker _gbm_create_device(int fd)
149*61046927SAndroid Build Coastguard Worker {
150*61046927SAndroid Build Coastguard Worker    struct gbm_device *dev = NULL;
151*61046927SAndroid Build Coastguard Worker 
152*61046927SAndroid Build Coastguard Worker    const char *b = getenv("GBM_BACKEND");
153*61046927SAndroid Build Coastguard Worker    if (b) {
154*61046927SAndroid Build Coastguard Worker       dev = load_backend_by_name(b, fd, true);
155*61046927SAndroid Build Coastguard Worker       if (dev) return dev;
156*61046927SAndroid Build Coastguard Worker    }
157*61046927SAndroid Build Coastguard Worker 
158*61046927SAndroid Build Coastguard Worker    drmVersionPtr v = drmGetVersion(fd);
159*61046927SAndroid Build Coastguard Worker    if (v) {
160*61046927SAndroid Build Coastguard Worker       dev = load_backend_by_name(v->name, fd, false);
161*61046927SAndroid Build Coastguard Worker       drmFreeVersion(v);
162*61046927SAndroid Build Coastguard Worker       if (dev) return dev;
163*61046927SAndroid Build Coastguard Worker    }
164*61046927SAndroid Build Coastguard Worker 
165*61046927SAndroid Build Coastguard Worker    return load_backend_by_name("dri", fd, true);
166*61046927SAndroid Build Coastguard Worker }
167*61046927SAndroid Build Coastguard Worker 
168*61046927SAndroid Build Coastguard Worker void
_gbm_device_destroy(struct gbm_device * gbm)169*61046927SAndroid Build Coastguard Worker _gbm_device_destroy(struct gbm_device *gbm)
170*61046927SAndroid Build Coastguard Worker {
171*61046927SAndroid Build Coastguard Worker    const struct gbm_backend_desc *backend_desc = gbm->v0.backend_desc;
172*61046927SAndroid Build Coastguard Worker    gbm->v0.destroy(gbm);
173*61046927SAndroid Build Coastguard Worker 
174*61046927SAndroid Build Coastguard Worker    if (backend_desc && backend_desc->lib)
175*61046927SAndroid Build Coastguard Worker       free_backend_desc(backend_desc);
176*61046927SAndroid Build Coastguard Worker }
177