mesa/src/gbm/main/backend.c
K900 514df444eb gbm: split the DRI backend into a separate library and unify backend handling
This does a few things:
1. Applications linked against a non-matching GBM can load the right version of the backend
   from the environment, avoiding GBM/Mesa version mismatches in many cases
2. Distros that want to split off libgbm into a separate package from the actual drivers
   get to not ship 40MB of libgallium_dri
3. The loader logic becomes way less complex

See also: https://github.com/NixOS/nixpkgs/pull/338109 for original rationale.

Part-of: <https://gitlab.freedesktop.org/mesa/mesa/-/merge_requests/31074>
2024-09-17 22:17:10 +00:00

176 lines
4.5 KiB
C

/*
* Copyright © 2011 Intel Corporation
* Copyright © 2021 NVIDIA Corporation
*
* 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 (including the next
* paragraph) 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 AUTHORS OR COPYRIGHT
* HOLDERS 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.
*
* Authors:
* Benjamin Franzke <benjaminfranzke@googlemail.com>
* James Jones <jajones@nvidia.com>
*/
#include <stdio.h>
#include <stddef.h>
#include <stdlib.h>
#include <string.h>
#include <limits.h>
#include <assert.h>
#include <dlfcn.h>
#include <xf86drm.h>
#include "loader.h"
#include "backend.h"
#define ARRAY_SIZE(a) (sizeof(a)/sizeof((a)[0]))
#define VER_MIN(a, b) ((a) < (b) ? (a) : (b))
struct gbm_backend_desc {
const char *name;
const struct gbm_backend *backend;
void *lib;
};
#define BACKEND_LIB_SUFFIX "_gbm"
static const char *backend_search_path_vars[] = {
"GBM_BACKENDS_PATH",
NULL
};
static void
free_backend_desc(const struct gbm_backend_desc *backend_desc)
{
assert(backend_desc->lib);
dlclose(backend_desc->lib);
free((void *)backend_desc->name);
free((void *)backend_desc);
}
static struct gbm_backend_desc *
create_backend_desc(const char *name,
const struct gbm_backend *backend,
void *lib)
{
struct gbm_backend_desc *new_desc = calloc(1, sizeof(*new_desc));
if (!new_desc)
return NULL;
new_desc->name = strdup(name);
if (!new_desc->name) {
free(new_desc);
return NULL;
}
new_desc->backend = backend;
new_desc->lib = lib;
return new_desc;
}
static struct gbm_device *
backend_create_device(const struct gbm_backend_desc *bd, int fd)
{
const uint32_t abi_ver = VER_MIN(GBM_BACKEND_ABI_VERSION,
bd->backend->v0.backend_version);
struct gbm_device *dev = bd->backend->v0.create_device(fd, abi_ver);
if (dev) {
if (abi_ver != dev->v0.backend_version) {
_gbm_device_destroy(dev);
return NULL;
}
dev->v0.backend_desc = bd;
}
return dev;
}
static struct gbm_device *
load_backend_by_name(const char *name, int fd, bool warn_on_fail)
{
void *lib = loader_open_driver_lib(name, BACKEND_LIB_SUFFIX,
backend_search_path_vars,
DEFAULT_BACKENDS_PATH,
warn_on_fail);
if (!lib)
return NULL;
struct gbm_device *dev = NULL;
struct gbm_backend_desc *backend_desc;
const struct gbm_backend *gbm_backend;
GBM_GET_BACKEND_PROC_PTR get_backend;
get_backend = dlsym(lib, GBM_GET_BACKEND_PROC_NAME);
if (!get_backend)
goto fail;
gbm_backend = get_backend(&gbm_core);
backend_desc = create_backend_desc(name, gbm_backend, lib);
if (!backend_desc)
goto fail;
dev = backend_create_device(backend_desc, fd);
if (!dev)
free_backend_desc(backend_desc);
return dev;
fail:
dlclose(lib);
return NULL;
}
struct gbm_device *
_gbm_create_device(int fd)
{
struct gbm_device *dev = NULL;
const char *b = getenv("GBM_BACKEND");
if (b) {
dev = load_backend_by_name(b, fd, true);
if (dev) return dev;
}
drmVersionPtr v = drmGetVersion(fd);
if (v) {
dev = load_backend_by_name(v->name, fd, false);
drmFreeVersion(v);
if (dev) return dev;
}
return load_backend_by_name("dri", fd, true);
}
void
_gbm_device_destroy(struct gbm_device *gbm)
{
const struct gbm_backend_desc *backend_desc = gbm->v0.backend_desc;
gbm->v0.destroy(gbm);
if (backend_desc && backend_desc->lib)
free_backend_desc(backend_desc);
}