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