gbm.c revision 3464ebd5
13464ebd5Sriastradh/*
23464ebd5Sriastradh * Copyright © 2011 Intel Corporation
33464ebd5Sriastradh *
43464ebd5Sriastradh * Permission is hereby granted, free of charge, to any person obtaining a
53464ebd5Sriastradh * copy of this software and associated documentation files (the "Software"),
63464ebd5Sriastradh * to deal in the Software without restriction, including without limitation
73464ebd5Sriastradh * the rights to use, copy, modify, merge, publish, distribute, sublicense,
83464ebd5Sriastradh * and/or sell copies of the Software, and to permit persons to whom the
93464ebd5Sriastradh * Software is furnished to do so, subject to the following conditions:
103464ebd5Sriastradh *
113464ebd5Sriastradh * The above copyright notice and this permission notice (including the next
123464ebd5Sriastradh * paragraph) shall be included in all copies or substantial portions of the
133464ebd5Sriastradh * Software.
143464ebd5Sriastradh *
153464ebd5Sriastradh * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
163464ebd5Sriastradh * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
173464ebd5Sriastradh * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
183464ebd5Sriastradh * NONINFRINGEMENT.  IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
193464ebd5Sriastradh * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
203464ebd5Sriastradh * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
213464ebd5Sriastradh * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
223464ebd5Sriastradh * DEALINGS IN THE SOFTWARE.
233464ebd5Sriastradh *
243464ebd5Sriastradh * Authors:
253464ebd5Sriastradh *    Benjamin Franzke <benjaminfranzke@googlemail.com>
263464ebd5Sriastradh */
273464ebd5Sriastradh
283464ebd5Sriastradh#define _BSD_SOURCE
293464ebd5Sriastradh
303464ebd5Sriastradh#include <stddef.h>
313464ebd5Sriastradh#include <stdio.h>
323464ebd5Sriastradh#include <stdlib.h>
333464ebd5Sriastradh#include <string.h>
343464ebd5Sriastradh#include <stdint.h>
353464ebd5Sriastradh
363464ebd5Sriastradh#include <sys/types.h>
373464ebd5Sriastradh#include <sys/stat.h>
383464ebd5Sriastradh#include <unistd.h>
393464ebd5Sriastradh
403464ebd5Sriastradh#include "gbm.h"
413464ebd5Sriastradh#include "gbmint.h"
423464ebd5Sriastradh#include "common.h"
433464ebd5Sriastradh#include "backend.h"
443464ebd5Sriastradh
453464ebd5Sriastradh#define ARRAY_SIZE(a) (sizeof(a)/sizeof((a)[0]))
463464ebd5Sriastradh
473464ebd5Sriastradhstruct gbm_device *devices[16];
483464ebd5Sriastradh
493464ebd5Sriastradhstatic int device_num = 0;
503464ebd5Sriastradh
513464ebd5SriastradhGBM_EXPORT int
523464ebd5Sriastradhgbm_device_get_fd(struct gbm_device *gbm)
533464ebd5Sriastradh{
543464ebd5Sriastradh   return gbm->fd;
553464ebd5Sriastradh}
563464ebd5Sriastradh
573464ebd5Sriastradh/* FIXME: maybe superfluous, use udev subclass from the fd? */
583464ebd5SriastradhGBM_EXPORT const char *
593464ebd5Sriastradhgbm_device_get_backend_name(struct gbm_device *gbm)
603464ebd5Sriastradh{
613464ebd5Sriastradh   return gbm->name;
623464ebd5Sriastradh}
633464ebd5Sriastradh
643464ebd5Sriastradhint
653464ebd5Sriastradhgbm_device_is_format_supported(struct gbm_device *gbm,
663464ebd5Sriastradh                               enum gbm_bo_format format,
673464ebd5Sriastradh                               uint32_t usage)
683464ebd5Sriastradh{
693464ebd5Sriastradh   return gbm->is_format_supported(gbm, format, usage);
703464ebd5Sriastradh}
713464ebd5Sriastradh
723464ebd5SriastradhGBM_EXPORT void
733464ebd5Sriastradhgbm_device_destroy(struct gbm_device *gbm)
743464ebd5Sriastradh{
753464ebd5Sriastradh   gbm->refcount--;
763464ebd5Sriastradh   if (gbm->refcount == 0)
773464ebd5Sriastradh      gbm->destroy(gbm);
783464ebd5Sriastradh}
793464ebd5Sriastradh
803464ebd5SriastradhGBM_EXPORT struct gbm_device *
813464ebd5Sriastradh_gbm_mesa_get_device(int fd)
823464ebd5Sriastradh{
833464ebd5Sriastradh   struct gbm_device *gbm = NULL;
843464ebd5Sriastradh   struct stat buf;
853464ebd5Sriastradh   dev_t dev;
863464ebd5Sriastradh   int i;
873464ebd5Sriastradh
883464ebd5Sriastradh   if (fd < 0 || fstat(fd, &buf) < 0 || !S_ISCHR(buf.st_mode)) {
893464ebd5Sriastradh      fprintf(stderr, "_gbm_mesa_get_device: invalid fd: %d\n", fd);
903464ebd5Sriastradh      return NULL;
913464ebd5Sriastradh   }
923464ebd5Sriastradh
933464ebd5Sriastradh   for (i = 0; i < device_num; ++i) {
943464ebd5Sriastradh      dev = devices[i]->stat.st_rdev;
953464ebd5Sriastradh      if (major(dev) == major(buf.st_rdev) &&
963464ebd5Sriastradh          minor(dev) == minor(buf.st_rdev)) {
973464ebd5Sriastradh         gbm = devices[i];
983464ebd5Sriastradh         gbm->refcount++;
993464ebd5Sriastradh         break;
1003464ebd5Sriastradh      }
1013464ebd5Sriastradh   }
1023464ebd5Sriastradh
1033464ebd5Sriastradh   return gbm;
1043464ebd5Sriastradh}
1053464ebd5Sriastradh
1063464ebd5SriastradhGBM_EXPORT struct gbm_device *
1073464ebd5Sriastradhgbm_create_device(int fd)
1083464ebd5Sriastradh{
1093464ebd5Sriastradh   struct gbm_device *gbm = NULL;
1103464ebd5Sriastradh   struct stat buf;
1113464ebd5Sriastradh
1123464ebd5Sriastradh   if (fd < 0 || fstat(fd, &buf) < 0 || !S_ISCHR(buf.st_mode)) {
1133464ebd5Sriastradh      fprintf(stderr, "gbm_create_device: invalid fd: %d\n", fd);
1143464ebd5Sriastradh      return NULL;
1153464ebd5Sriastradh   }
1163464ebd5Sriastradh
1173464ebd5Sriastradh   if (device_num == 0)
1183464ebd5Sriastradh      memset(devices, 0, sizeof devices);
1193464ebd5Sriastradh
1203464ebd5Sriastradh   gbm = _gbm_create_device(fd);
1213464ebd5Sriastradh   if (gbm == NULL)
1223464ebd5Sriastradh      return NULL;
1233464ebd5Sriastradh
1243464ebd5Sriastradh   gbm->dummy = gbm_create_device;
1253464ebd5Sriastradh   gbm->stat = buf;
1263464ebd5Sriastradh   gbm->refcount = 1;
1273464ebd5Sriastradh
1283464ebd5Sriastradh   if (device_num < ARRAY_SIZE(devices)-1)
1293464ebd5Sriastradh      devices[device_num++] = gbm;
1303464ebd5Sriastradh
1313464ebd5Sriastradh   return gbm;
1323464ebd5Sriastradh}
1333464ebd5Sriastradh
1343464ebd5SriastradhGBM_EXPORT unsigned int
1353464ebd5Sriastradhgbm_bo_get_width(struct gbm_bo *bo)
1363464ebd5Sriastradh{
1373464ebd5Sriastradh   return bo->width;
1383464ebd5Sriastradh}
1393464ebd5Sriastradh
1403464ebd5SriastradhGBM_EXPORT unsigned int
1413464ebd5Sriastradhgbm_bo_get_height(struct gbm_bo *bo)
1423464ebd5Sriastradh{
1433464ebd5Sriastradh   return bo->height;
1443464ebd5Sriastradh}
1453464ebd5Sriastradh
1463464ebd5SriastradhGBM_EXPORT uint32_t
1473464ebd5Sriastradhgbm_bo_get_pitch(struct gbm_bo *bo)
1483464ebd5Sriastradh{
1493464ebd5Sriastradh   return bo->pitch;
1503464ebd5Sriastradh}
1513464ebd5Sriastradh
1523464ebd5SriastradhGBM_EXPORT union gbm_bo_handle
1533464ebd5Sriastradhgbm_bo_get_handle(struct gbm_bo *bo)
1543464ebd5Sriastradh{
1553464ebd5Sriastradh   return bo->handle;
1563464ebd5Sriastradh}
1573464ebd5Sriastradh
1583464ebd5SriastradhGBM_EXPORT void
1593464ebd5Sriastradhgbm_bo_destroy(struct gbm_bo *bo)
1603464ebd5Sriastradh{
1613464ebd5Sriastradh   bo->gbm->bo_destroy(bo);
1623464ebd5Sriastradh}
1633464ebd5Sriastradh
1643464ebd5SriastradhGBM_EXPORT struct gbm_bo *
1653464ebd5Sriastradhgbm_bo_create(struct gbm_device *gbm,
1663464ebd5Sriastradh              uint32_t width, uint32_t height,
1673464ebd5Sriastradh              enum gbm_bo_format format, uint32_t usage)
1683464ebd5Sriastradh{
1693464ebd5Sriastradh   if (width == 0 || height == 0)
1703464ebd5Sriastradh      return NULL;
1713464ebd5Sriastradh
1723464ebd5Sriastradh   if (usage & GBM_BO_USE_CURSOR_64X64 &&
1733464ebd5Sriastradh       (width != 64 || height != 64))
1743464ebd5Sriastradh      return NULL;
1753464ebd5Sriastradh
1763464ebd5Sriastradh   return gbm->bo_create(gbm, width, height, format, usage);
1773464ebd5Sriastradh}
1783464ebd5Sriastradh
1793464ebd5SriastradhGBM_EXPORT struct gbm_bo *
1803464ebd5Sriastradhgbm_bo_create_from_egl_image(struct gbm_device *gbm,
1813464ebd5Sriastradh                             void *egl_dpy, void *egl_image,
1823464ebd5Sriastradh                             uint32_t width, uint32_t height,
1833464ebd5Sriastradh                             uint32_t usage)
1843464ebd5Sriastradh{
1853464ebd5Sriastradh   if (width == 0 || height == 0)
1863464ebd5Sriastradh      return NULL;
1873464ebd5Sriastradh
1883464ebd5Sriastradh   return gbm->bo_create_from_egl_image(gbm, egl_dpy, egl_image,
1893464ebd5Sriastradh                                        width, height, usage);
1903464ebd5Sriastradh}
191