/* * Copyright (C) 2009 Francisco Jerez. * All Rights Reserved. * * 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 COPYRIGHT OWNER(S) AND/OR ITS SUPPLIERS 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. * */ #include <stdio.h> #include <xf86drm.h> #include <nouveau_drm.h> #include "nouveau_driver.h" #include "nouveau_context.h" #include "nouveau_fbo.h" #include "nouveau_texture.h" #include "nv04_driver.h" #include "nv10_driver.h" #include "nv20_driver.h" #include "main/framebuffer.h" #include "main/fbobject.h" #include "main/renderbuffer.h" #include "swrast/s_renderbuffer.h" #include <nvif/class.h> #include <nvif/cl0080.h> static const __DRIextension *nouveau_screen_extensions[]; static void nouveau_destroy_screen(__DRIscreen *dri_screen); static const __DRIconfig ** nouveau_get_configs(uint32_t chipset) { __DRIconfig **configs = NULL; int i; const uint8_t depth_bits[] = { 0, 16, 24, 24 }; const uint8_t stencil_bits[] = { 0, 0, 0, 8 }; const uint8_t msaa_samples[] = { 0 }; static const mesa_format formats[3] = { MESA_FORMAT_B5G6R5_UNORM, MESA_FORMAT_B8G8R8A8_UNORM, MESA_FORMAT_B8G8R8X8_UNORM, }; const GLenum back_buffer_modes[] = { GLX_NONE, GLX_SWAP_UNDEFINED_OML }; for (i = 0; i < ARRAY_SIZE(formats); i++) { __DRIconfig **config; config = driCreateConfigs(formats[i], depth_bits, stencil_bits, ARRAY_SIZE(depth_bits), back_buffer_modes, ARRAY_SIZE(back_buffer_modes), msaa_samples, ARRAY_SIZE(msaa_samples), GL_TRUE, chipset < 0x10); assert(config); configs = driConcatConfigs(configs, config); } return (const __DRIconfig **)configs; } static const __DRIconfig ** nouveau_init_screen2(__DRIscreen *dri_screen) { const __DRIconfig **configs; struct nouveau_screen *screen; int ret; /* Allocate the screen. */ screen = CALLOC_STRUCT(nouveau_screen); if (!screen) return NULL; dri_screen->driverPrivate = screen; /* Open the DRM device. */ ret = nouveau_drm_new(dri_screen->fd, &screen->drm); if (ret) { nouveau_error("Error opening the DRM device.\n"); goto fail; } ret = nouveau_device_new(&screen->drm->client, NV_DEVICE, &(struct nv_device_v0) { .device = ~0ULL, }, sizeof(struct nv_device_v0), &screen->device); if (ret) { nouveau_error("Error creating device object.\n"); goto fail; } /* Choose the card specific function pointers. */ switch (screen->device->chipset & 0xf0) { case 0x00: screen->driver = &nv04_driver; dri_screen->max_gl_compat_version = 12; break; case 0x10: screen->driver = &nv10_driver; dri_screen->max_gl_compat_version = 12; dri_screen->max_gl_es1_version = 10; break; case 0x20: case 0x30: screen->driver = &nv20_driver; dri_screen->max_gl_compat_version = 13; dri_screen->max_gl_es1_version = 10; break; default: nouveau_error("Unknown chipset: %02X\n", screen->device->chipset); goto fail; } dri_screen->extensions = nouveau_screen_extensions; screen->dri_screen = dri_screen; configs = nouveau_get_configs(screen->device->chipset); if (!configs) goto fail; return configs; fail: nouveau_destroy_screen(dri_screen); return NULL; } static int nouveau_query_renderer_integer(__DRIscreen *psp, int param, unsigned int *value) { const struct nouveau_screen *const screen = (struct nouveau_screen *) psp->driverPrivate; switch (param) { case __DRI2_RENDERER_VENDOR_ID: value[0] = 0x10de; return 0; case __DRI2_RENDERER_DEVICE_ID: { uint64_t device_id; if (nouveau_getparam(screen->device, NOUVEAU_GETPARAM_PCI_DEVICE, &device_id)) { nouveau_error("Error retrieving the device PCIID.\n"); device_id = -1; } value[0] = (unsigned int) device_id; return 0; } case __DRI2_RENDERER_ACCELERATED: value[0] = 1; return 0; case __DRI2_RENDERER_VIDEO_MEMORY: /* XXX: return vram_size or vram_limit ? */ value[0] = screen->device->vram_size >> 20; return 0; case __DRI2_RENDERER_UNIFIED_MEMORY_ARCHITECTURE: value[0] = 0; return 0; default: return driQueryRendererIntegerCommon(psp, param, value); } } static int nouveau_query_renderer_string(__DRIscreen *psp, int param, const char **value) { const struct nouveau_screen *const screen = (struct nouveau_screen *) psp->driverPrivate; switch (param) { case __DRI2_RENDERER_VENDOR_ID: value[0] = nouveau_vendor_string; return 0; case __DRI2_RENDERER_DEVICE_ID: value[0] = nouveau_get_renderer_string(screen->device->chipset); return 0; default: return -1; } } static const __DRI2rendererQueryExtension nouveau_renderer_query_extension = { .base = { __DRI2_RENDERER_QUERY, 1 }, .queryInteger = nouveau_query_renderer_integer, .queryString = nouveau_query_renderer_string }; static void nouveau_destroy_screen(__DRIscreen *dri_screen) { struct nouveau_screen *screen = dri_screen->driverPrivate; if (!screen) return; nouveau_device_del(&screen->device); nouveau_drm_del(&screen->drm); free(screen); dri_screen->driverPrivate = NULL; } static GLboolean nouveau_create_buffer(__DRIscreen *dri_screen, __DRIdrawable *drawable, const struct gl_config *visual, GLboolean is_pixmap) { struct gl_renderbuffer *rb; struct gl_framebuffer *fb; GLenum color_format; if (is_pixmap) return GL_FALSE; /* not implemented */ if (visual->redBits == 5) color_format = GL_RGB5; else if (visual->alphaBits == 0) color_format = GL_RGB8; else color_format = GL_RGBA8; fb = nouveau_framebuffer_dri_new(visual); if (!fb) return GL_FALSE; /* Front buffer. */ rb = nouveau_renderbuffer_dri_new(color_format, drawable); _mesa_add_renderbuffer(fb, BUFFER_FRONT_LEFT, rb); /* Back buffer */ if (visual->doubleBufferMode) { rb = nouveau_renderbuffer_dri_new(color_format, drawable); _mesa_add_renderbuffer(fb, BUFFER_BACK_LEFT, rb); } /* Depth/stencil buffer. */ if (visual->depthBits == 24 && visual->stencilBits == 8) { rb = nouveau_renderbuffer_dri_new(GL_DEPTH24_STENCIL8_EXT, drawable); _mesa_add_renderbuffer(fb, BUFFER_DEPTH, rb); _mesa_add_renderbuffer(fb, BUFFER_STENCIL, rb); } else if (visual->depthBits == 24) { rb = nouveau_renderbuffer_dri_new(GL_DEPTH_COMPONENT24, drawable); _mesa_add_renderbuffer(fb, BUFFER_DEPTH, rb); } else if (visual->depthBits == 16) { rb = nouveau_renderbuffer_dri_new(GL_DEPTH_COMPONENT16, drawable); _mesa_add_renderbuffer(fb, BUFFER_DEPTH, rb); } /* Software renderbuffers. */ _swrast_add_soft_renderbuffers(fb, GL_FALSE, GL_FALSE, GL_FALSE, visual->accumRedBits > 0, GL_FALSE, GL_FALSE); drawable->driverPrivate = fb; return GL_TRUE; } static void nouveau_destroy_buffer(__DRIdrawable *drawable) { _mesa_reference_framebuffer( (struct gl_framebuffer **)&drawable->driverPrivate, NULL); } static void nouveau_drawable_flush(__DRIdrawable *draw) { } static const struct __DRI2flushExtensionRec nouveau_flush_extension = { .base = { __DRI2_FLUSH, 3 }, .flush = nouveau_drawable_flush, .invalidate = dri2InvalidateDrawable, }; static const struct __DRItexBufferExtensionRec nouveau_texbuffer_extension = { .base = { __DRI_TEX_BUFFER, 3 }, .setTexBuffer = NULL, .setTexBuffer2 = nouveau_set_texbuffer, .releaseTexBuffer = NULL, }; static const __DRIextension *nouveau_screen_extensions[] = { &nouveau_flush_extension.base, &nouveau_texbuffer_extension.base, &nouveau_renderer_query_extension.base, &dri2ConfigQueryExtension.base, NULL }; const struct __DriverAPIRec nouveau_driver_api = { .InitScreen = nouveau_init_screen2, .DestroyScreen = nouveau_destroy_screen, .CreateBuffer = nouveau_create_buffer, .DestroyBuffer = nouveau_destroy_buffer, .CreateContext = nouveau_context_create, .DestroyContext = nouveau_context_destroy, .MakeCurrent = nouveau_context_make_current, .UnbindContext = nouveau_context_unbind, }; static const struct __DRIDriverVtableExtensionRec nouveau_vtable = { .base = { __DRI_DRIVER_VTABLE, 1 }, .vtable = &nouveau_driver_api, }; /* This is the table of extensions that the loader will dlsym() for. */ static const __DRIextension *nouveau_driver_extensions[] = { &driCoreExtension.base, &driDRI2Extension.base, &nouveau_vtable.base, NULL }; PUBLIC const __DRIextension **__driDriverGetExtensions_nouveau_vieux(void) { globalDriverAPI = &nouveau_driver_api; return nouveau_driver_extensions; }