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