/* * Copyright 2010 Red Hat Inc. * * 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 * on the rights to use, copy, modify, merge, publish, distribute, sub * license, 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 NON-INFRINGEMENT. IN NO EVENT SHALL * THE AUTHOR(S) AND/OR THEIR 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 #include #include "pipe/p_defines.h" #include "pipe/p_state.h" #include "pipe/p_context.h" #include "pipe/p_screen.h" #include "util/u_memory.h" #include "util/u_inlines.h" #include "util/format/u_format.h" #include "util/u_helpers.h" #include "util/u_upload_mgr.h" #include "util/u_threaded_context.h" #include "noop_public.h" DEBUG_GET_ONCE_BOOL_OPTION(noop, "GALLIUM_NOOP", false) void noop_init_state_functions(struct pipe_context *ctx); struct noop_pipe_screen { struct pipe_screen pscreen; struct pipe_screen *oscreen; struct slab_parent_pool pool_transfers; }; /* * query */ struct noop_query { struct threaded_query b; unsigned query; }; static struct pipe_query *noop_create_query(struct pipe_context *ctx, unsigned query_type, unsigned index) { struct noop_query *query = CALLOC_STRUCT(noop_query); return (struct pipe_query *)query; } static void noop_destroy_query(struct pipe_context *ctx, struct pipe_query *query) { FREE(query); } static bool noop_begin_query(struct pipe_context *ctx, struct pipe_query *query) { return true; } static bool noop_end_query(struct pipe_context *ctx, struct pipe_query *query) { return true; } static bool noop_get_query_result(struct pipe_context *ctx, struct pipe_query *query, bool wait, union pipe_query_result *vresult) { uint64_t *result = (uint64_t*)vresult; *result = 0; return true; } static void noop_set_active_query_state(struct pipe_context *pipe, bool enable) { } /* * resource */ struct noop_resource { struct threaded_resource b; unsigned size; char *data; struct sw_displaytarget *dt; }; static struct pipe_resource *noop_resource_create(struct pipe_screen *screen, const struct pipe_resource *templ) { struct noop_resource *nresource; unsigned stride; nresource = CALLOC_STRUCT(noop_resource); if (!nresource) return NULL; stride = util_format_get_stride(templ->format, templ->width0); nresource->b.b = *templ; nresource->b.b.screen = screen; nresource->size = stride * templ->height0 * templ->depth0; nresource->data = MALLOC(nresource->size); pipe_reference_init(&nresource->b.b.reference, 1); if (nresource->data == NULL) { FREE(nresource); return NULL; } threaded_resource_init(&nresource->b.b); return &nresource->b.b; } static struct pipe_resource * noop_resource_create_with_modifiers(struct pipe_screen *screen, const struct pipe_resource *templ, const uint64_t *modifiers, int count) { struct noop_pipe_screen *noop_screen = (struct noop_pipe_screen*)screen; struct pipe_screen *oscreen = noop_screen->oscreen; struct pipe_resource *result; struct pipe_resource *noop_resource; result = oscreen->resource_create_with_modifiers(oscreen, templ, modifiers, count); noop_resource = noop_resource_create(screen, result); pipe_resource_reference(&result, NULL); return noop_resource; } static struct pipe_resource *noop_resource_from_handle(struct pipe_screen *screen, const struct pipe_resource *templ, struct winsys_handle *handle, unsigned usage) { struct noop_pipe_screen *noop_screen = (struct noop_pipe_screen*)screen; struct pipe_screen *oscreen = noop_screen->oscreen; struct pipe_resource *result; struct pipe_resource *noop_resource; result = oscreen->resource_from_handle(oscreen, templ, handle, usage); noop_resource = noop_resource_create(screen, result); pipe_resource_reference(&result, NULL); return noop_resource; } static bool noop_resource_get_handle(struct pipe_screen *pscreen, struct pipe_context *ctx, struct pipe_resource *resource, struct winsys_handle *handle, unsigned usage) { struct noop_pipe_screen *noop_screen = (struct noop_pipe_screen*)pscreen; struct pipe_screen *screen = noop_screen->oscreen; struct pipe_resource *tex; bool result; /* resource_get_handle musn't fail. Just create something and return it. */ tex = screen->resource_create(screen, resource); if (!tex) return false; result = screen->resource_get_handle(screen, NULL, tex, handle, usage); pipe_resource_reference(&tex, NULL); return result; } static bool noop_resource_get_param(struct pipe_screen *pscreen, struct pipe_context *ctx, struct pipe_resource *resource, unsigned plane, unsigned layer, unsigned level, enum pipe_resource_param param, unsigned handle_usage, uint64_t *value) { struct noop_pipe_screen *noop_screen = (struct noop_pipe_screen*)pscreen; struct pipe_screen *screen = noop_screen->oscreen; struct pipe_resource *tex; bool result; /* resource_get_param mustn't fail. Just create something and return it. */ tex = screen->resource_create(screen, resource); if (!tex) return false; result = screen->resource_get_param(screen, NULL, tex, 0, 0, 0, param, handle_usage, value); pipe_resource_reference(&tex, NULL); return result; } static void noop_resource_destroy(struct pipe_screen *screen, struct pipe_resource *resource) { struct noop_resource *nresource = (struct noop_resource *)resource; threaded_resource_deinit(resource); FREE(nresource->data); FREE(resource); } /* * transfer */ static void *noop_transfer_map(struct pipe_context *pipe, struct pipe_resource *resource, unsigned level, unsigned usage, const struct pipe_box *box, struct pipe_transfer **ptransfer) { struct pipe_transfer *transfer; struct noop_resource *nresource = (struct noop_resource *)resource; transfer = (struct pipe_transfer*)CALLOC_STRUCT(threaded_transfer); if (!transfer) return NULL; pipe_resource_reference(&transfer->resource, resource); transfer->level = level; transfer->usage = usage; transfer->box = *box; transfer->stride = 1; transfer->layer_stride = 1; *ptransfer = transfer; return nresource->data; } static void noop_transfer_flush_region(struct pipe_context *pipe, struct pipe_transfer *transfer, const struct pipe_box *box) { } static void noop_transfer_unmap(struct pipe_context *pipe, struct pipe_transfer *transfer) { pipe_resource_reference(&transfer->resource, NULL); FREE(transfer); } static void noop_buffer_subdata(struct pipe_context *pipe, struct pipe_resource *resource, unsigned usage, unsigned offset, unsigned size, const void *data) { } static void noop_texture_subdata(struct pipe_context *pipe, struct pipe_resource *resource, unsigned level, unsigned usage, const struct pipe_box *box, const void *data, unsigned stride, unsigned layer_stride) { } /* * clear/copy */ static void noop_clear(struct pipe_context *ctx, unsigned buffers, const struct pipe_scissor_state *scissor_state, const union pipe_color_union *color, double depth, unsigned stencil) { } static void noop_clear_render_target(struct pipe_context *ctx, struct pipe_surface *dst, const union pipe_color_union *color, unsigned dstx, unsigned dsty, unsigned width, unsigned height, bool render_condition_enabled) { } static void noop_clear_depth_stencil(struct pipe_context *ctx, struct pipe_surface *dst, unsigned clear_flags, double depth, unsigned stencil, unsigned dstx, unsigned dsty, unsigned width, unsigned height, bool render_condition_enabled) { } static void noop_resource_copy_region(struct pipe_context *ctx, struct pipe_resource *dst, unsigned dst_level, unsigned dstx, unsigned dsty, unsigned dstz, struct pipe_resource *src, unsigned src_level, const struct pipe_box *src_box) { } static void noop_blit(struct pipe_context *ctx, const struct pipe_blit_info *info) { } static void noop_flush_resource(struct pipe_context *ctx, struct pipe_resource *resource) { } /* * context */ static void noop_flush(struct pipe_context *ctx, struct pipe_fence_handle **fence, unsigned flags) { if (fence) { struct pipe_reference *f = MALLOC_STRUCT(pipe_reference); f->count = 1; ctx->screen->fence_reference(ctx->screen, fence, NULL); *fence = (struct pipe_fence_handle*)f; } } static void noop_destroy_context(struct pipe_context *ctx) { if (ctx->stream_uploader) u_upload_destroy(ctx->stream_uploader); p_atomic_dec(&ctx->screen->num_contexts); FREE(ctx); } static bool noop_generate_mipmap(struct pipe_context *ctx, struct pipe_resource *resource, enum pipe_format format, unsigned base_level, unsigned last_level, unsigned first_layer, unsigned last_layer) { return true; } static void noop_invalidate_resource(struct pipe_context *ctx, struct pipe_resource *resource) { } static void noop_set_context_param(struct pipe_context *ctx, enum pipe_context_param param, unsigned value) { } static void noop_set_frontend_noop(struct pipe_context *ctx, bool enable) { } static void noop_replace_buffer_storage(struct pipe_context *ctx, struct pipe_resource *dst, struct pipe_resource *src, unsigned num_rebinds, uint32_t rebind_mask, uint32_t delete_buffer_id) { } static struct pipe_fence_handle * noop_create_fence(struct pipe_context *ctx, struct tc_unflushed_batch_token *tc_token) { struct pipe_reference *f = MALLOC_STRUCT(pipe_reference); f->count = 1; return (struct pipe_fence_handle*)f; } static bool noop_is_resource_busy(struct pipe_screen *screen, struct pipe_resource *resource, unsigned usage) { return false; } static struct pipe_context *noop_create_context(struct pipe_screen *screen, void *priv, unsigned flags) { struct pipe_context *ctx = CALLOC_STRUCT(pipe_context); if (!ctx) return NULL; ctx->screen = screen; ctx->priv = priv; ctx->stream_uploader = u_upload_create_default(ctx); if (!ctx->stream_uploader) { FREE(ctx); return NULL; } ctx->const_uploader = ctx->stream_uploader; ctx->destroy = noop_destroy_context; ctx->flush = noop_flush; ctx->clear = noop_clear; ctx->clear_render_target = noop_clear_render_target; ctx->clear_depth_stencil = noop_clear_depth_stencil; ctx->resource_copy_region = noop_resource_copy_region; ctx->generate_mipmap = noop_generate_mipmap; ctx->blit = noop_blit; ctx->flush_resource = noop_flush_resource; ctx->create_query = noop_create_query; ctx->destroy_query = noop_destroy_query; ctx->begin_query = noop_begin_query; ctx->end_query = noop_end_query; ctx->get_query_result = noop_get_query_result; ctx->set_active_query_state = noop_set_active_query_state; ctx->buffer_map = noop_transfer_map; ctx->texture_map = noop_transfer_map; ctx->transfer_flush_region = noop_transfer_flush_region; ctx->buffer_unmap = noop_transfer_unmap; ctx->texture_unmap = noop_transfer_unmap; ctx->buffer_subdata = noop_buffer_subdata; ctx->texture_subdata = noop_texture_subdata; ctx->invalidate_resource = noop_invalidate_resource; ctx->set_context_param = noop_set_context_param; ctx->set_frontend_noop = noop_set_frontend_noop; noop_init_state_functions(ctx); p_atomic_inc(&screen->num_contexts); if (!(flags & PIPE_CONTEXT_PREFER_THREADED)) return ctx; struct pipe_context *tc = threaded_context_create(ctx, &((struct noop_pipe_screen*)screen)->pool_transfers, noop_replace_buffer_storage, &(struct threaded_context_options) { .create_fence = noop_create_fence, .is_resource_busy = noop_is_resource_busy, }, NULL); if (tc && tc != ctx) threaded_context_init_bytes_mapped_limit((struct threaded_context *)tc, 4); return tc; } /* * pipe_screen */ static void noop_flush_frontbuffer(struct pipe_screen *_screen, struct pipe_context *ctx, struct pipe_resource *resource, unsigned level, unsigned layer, void *context_private, struct pipe_box *box) { } static const char *noop_get_vendor(struct pipe_screen* pscreen) { return "X.Org"; } static const char *noop_get_device_vendor(struct pipe_screen* pscreen) { return "NONE"; } static const char *noop_get_name(struct pipe_screen* pscreen) { return "NOOP"; } static int noop_get_param(struct pipe_screen* pscreen, enum pipe_cap param) { struct pipe_screen *screen = ((struct noop_pipe_screen*)pscreen)->oscreen; return screen->get_param(screen, param); } static float noop_get_paramf(struct pipe_screen* pscreen, enum pipe_capf param) { struct pipe_screen *screen = ((struct noop_pipe_screen*)pscreen)->oscreen; return screen->get_paramf(screen, param); } static int noop_get_shader_param(struct pipe_screen* pscreen, enum pipe_shader_type shader, enum pipe_shader_cap param) { struct pipe_screen *screen = ((struct noop_pipe_screen*)pscreen)->oscreen; return screen->get_shader_param(screen, shader, param); } static int noop_get_compute_param(struct pipe_screen *pscreen, enum pipe_shader_ir ir_type, enum pipe_compute_cap param, void *ret) { struct pipe_screen *screen = ((struct noop_pipe_screen*)pscreen)->oscreen; return screen->get_compute_param(screen, ir_type, param, ret); } static bool noop_is_format_supported(struct pipe_screen* pscreen, enum pipe_format format, enum pipe_texture_target target, unsigned sample_count, unsigned storage_sample_count, unsigned usage) { struct pipe_screen *screen = ((struct noop_pipe_screen*)pscreen)->oscreen; return screen->is_format_supported(screen, format, target, sample_count, storage_sample_count, usage); } static uint64_t noop_get_timestamp(struct pipe_screen *pscreen) { return 0; } static void noop_destroy_screen(struct pipe_screen *screen) { struct noop_pipe_screen *noop_screen = (struct noop_pipe_screen*)screen; struct pipe_screen *oscreen = noop_screen->oscreen; oscreen->destroy(oscreen); slab_destroy_parent(&noop_screen->pool_transfers); FREE(screen); } static void noop_fence_reference(struct pipe_screen *screen, struct pipe_fence_handle **ptr, struct pipe_fence_handle *fence) { if (pipe_reference((struct pipe_reference*)*ptr, (struct pipe_reference*)fence)) FREE(*ptr); *ptr = fence; } static bool noop_fence_finish(struct pipe_screen *screen, struct pipe_context *ctx, struct pipe_fence_handle *fence, uint64_t timeout) { return true; } static void noop_query_memory_info(struct pipe_screen *pscreen, struct pipe_memory_info *info) { struct noop_pipe_screen *noop_screen = (struct noop_pipe_screen*)pscreen; struct pipe_screen *screen = noop_screen->oscreen; screen->query_memory_info(screen, info); } static struct disk_cache *noop_get_disk_shader_cache(struct pipe_screen *pscreen) { struct pipe_screen *screen = ((struct noop_pipe_screen*)pscreen)->oscreen; return screen->get_disk_shader_cache(screen); } static const void *noop_get_compiler_options(struct pipe_screen *pscreen, enum pipe_shader_ir ir, enum pipe_shader_type shader) { struct pipe_screen *screen = ((struct noop_pipe_screen*)pscreen)->oscreen; return screen->get_compiler_options(screen, ir, shader); } static char *noop_finalize_nir(struct pipe_screen *pscreen, void *nir) { struct pipe_screen *screen = ((struct noop_pipe_screen*)pscreen)->oscreen; return screen->finalize_nir(screen, nir); } static bool noop_check_resource_capability(struct pipe_screen *screen, struct pipe_resource *resource, unsigned bind) { return true; } static void noop_set_max_shader_compiler_threads(struct pipe_screen *screen, unsigned max_threads) { } static bool noop_is_parallel_shader_compilation_finished(struct pipe_screen *screen, void *shader, unsigned shader_type) { return true; } static bool noop_is_dmabuf_modifier_supported(struct pipe_screen *screen, uint64_t modifier, enum pipe_format format, bool *external_only) { struct noop_pipe_screen *noop_screen = (struct noop_pipe_screen*)screen; struct pipe_screen *oscreen = noop_screen->oscreen; return oscreen->is_dmabuf_modifier_supported(oscreen, modifier, format, external_only); } static unsigned int noop_get_dmabuf_modifier_planes(struct pipe_screen *screen, uint64_t modifier, enum pipe_format format) { struct noop_pipe_screen *noop_screen = (struct noop_pipe_screen*)screen; struct pipe_screen *oscreen = noop_screen->oscreen; return oscreen->get_dmabuf_modifier_planes(oscreen, modifier, format); } static void noop_get_driver_uuid(struct pipe_screen *screen, char *uuid) { struct noop_pipe_screen *noop_screen = (struct noop_pipe_screen*)screen; struct pipe_screen *oscreen = noop_screen->oscreen; oscreen->get_driver_uuid(oscreen, uuid); } static void noop_get_device_uuid(struct pipe_screen *screen, char *uuid) { struct noop_pipe_screen *noop_screen = (struct noop_pipe_screen*)screen; struct pipe_screen *oscreen = noop_screen->oscreen; oscreen->get_device_uuid(oscreen, uuid); } static void noop_query_dmabuf_modifiers(struct pipe_screen *screen, enum pipe_format format, int max, uint64_t *modifiers, unsigned int *external_only, int *count) { struct noop_pipe_screen *noop_screen = (struct noop_pipe_screen*)screen; struct pipe_screen *oscreen = noop_screen->oscreen; oscreen->query_dmabuf_modifiers(oscreen, format, max, modifiers, external_only, count); } static struct pipe_vertex_state * noop_create_vertex_state(struct pipe_screen *screen, struct pipe_vertex_buffer *buffer, const struct pipe_vertex_element *elements, unsigned num_elements, struct pipe_resource *indexbuf, uint32_t full_velem_mask) { struct pipe_vertex_state *state = CALLOC_STRUCT(pipe_vertex_state); if (!state) return NULL; util_init_pipe_vertex_state(screen, buffer, elements, num_elements, indexbuf, full_velem_mask, state); return state; } static void noop_vertex_state_destroy(struct pipe_screen *screen, struct pipe_vertex_state *state) { pipe_vertex_buffer_unreference(&state->input.vbuffer); pipe_resource_reference(&state->input.indexbuf, NULL); FREE(state); } struct pipe_screen *noop_screen_create(struct pipe_screen *oscreen) { struct noop_pipe_screen *noop_screen; struct pipe_screen *screen; if (!debug_get_option_noop()) { return oscreen; } noop_screen = CALLOC_STRUCT(noop_pipe_screen); if (!noop_screen) { return NULL; } noop_screen->oscreen = oscreen; screen = &noop_screen->pscreen; screen->destroy = noop_destroy_screen; screen->get_name = noop_get_name; screen->get_vendor = noop_get_vendor; screen->get_device_vendor = noop_get_device_vendor; screen->get_param = noop_get_param; screen->get_shader_param = noop_get_shader_param; screen->get_compute_param = noop_get_compute_param; screen->get_paramf = noop_get_paramf; screen->is_format_supported = noop_is_format_supported; screen->context_create = noop_create_context; screen->resource_create = noop_resource_create; screen->resource_from_handle = noop_resource_from_handle; screen->resource_get_handle = noop_resource_get_handle; if (oscreen->resource_get_param) screen->resource_get_param = noop_resource_get_param; screen->resource_destroy = noop_resource_destroy; screen->flush_frontbuffer = noop_flush_frontbuffer; screen->get_timestamp = noop_get_timestamp; screen->fence_reference = noop_fence_reference; screen->fence_finish = noop_fence_finish; screen->query_memory_info = noop_query_memory_info; screen->get_disk_shader_cache = noop_get_disk_shader_cache; screen->get_compiler_options = noop_get_compiler_options; screen->finalize_nir = noop_finalize_nir; screen->check_resource_capability = noop_check_resource_capability; screen->set_max_shader_compiler_threads = noop_set_max_shader_compiler_threads; screen->is_parallel_shader_compilation_finished = noop_is_parallel_shader_compilation_finished; screen->is_dmabuf_modifier_supported = noop_is_dmabuf_modifier_supported; screen->get_dmabuf_modifier_planes = noop_get_dmabuf_modifier_planes; screen->get_driver_uuid = noop_get_driver_uuid; screen->get_device_uuid = noop_get_device_uuid; screen->query_dmabuf_modifiers = noop_query_dmabuf_modifiers; screen->resource_create_with_modifiers = noop_resource_create_with_modifiers; screen->create_vertex_state = noop_create_vertex_state; screen->vertex_state_destroy = noop_vertex_state_destroy; slab_create_parent(&noop_screen->pool_transfers, sizeof(struct pipe_transfer), 64); return screen; }