17ec681f3Smrg/*
27ec681f3Smrg * Copyright 2011 Joakim Sindholt <opensource@zhasha.com>
37ec681f3Smrg * Copyright 2015 Patrick Rudolph <siro@das-labor.org>
47ec681f3Smrg *
57ec681f3Smrg * Permission is hereby granted, free of charge, to any person obtaining a
67ec681f3Smrg * copy of this software and associated documentation files (the "Software"),
77ec681f3Smrg * to deal in the Software without restriction, including without limitation
87ec681f3Smrg * on the rights to use, copy, modify, merge, publish, distribute, sub
97ec681f3Smrg * license, and/or sell copies of the Software, and to permit persons to whom
107ec681f3Smrg * the Software is furnished to do so, subject to the following conditions:
117ec681f3Smrg *
127ec681f3Smrg * The above copyright notice and this permission notice (including the next
137ec681f3Smrg * paragraph) shall be included in all copies or substantial portions of the
147ec681f3Smrg * Software.
157ec681f3Smrg *
167ec681f3Smrg * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
177ec681f3Smrg * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
187ec681f3Smrg * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL
197ec681f3Smrg * THE AUTHOR(S) AND/OR THEIR SUPPLIERS BE LIABLE FOR ANY CLAIM,
207ec681f3Smrg * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
217ec681f3Smrg * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
227ec681f3Smrg * USE OR OTHER DEALINGS IN THE SOFTWARE. */
237ec681f3Smrg
247ec681f3Smrg#ifndef _NINE_BUFFER9_H_
257ec681f3Smrg#define _NINE_BUFFER9_H_
267ec681f3Smrg
277ec681f3Smrg#include "device9.h"
287ec681f3Smrg#include "nine_buffer_upload.h"
297ec681f3Smrg#include "nine_state.h"
307ec681f3Smrg#include "resource9.h"
317ec681f3Smrg#include "pipe/p_context.h"
327ec681f3Smrg#include "pipe/p_defines.h"
337ec681f3Smrg#include "pipe/p_state.h"
347ec681f3Smrg#include "util/list.h"
357ec681f3Smrg#include "util/u_box.h"
367ec681f3Smrg#include "util/u_upload_mgr.h"
377ec681f3Smrg
387ec681f3Smrgstruct pipe_screen;
397ec681f3Smrgstruct pipe_context;
407ec681f3Smrgstruct pipe_transfer;
417ec681f3Smrg
427ec681f3Smrgstruct NineTransfer {
437ec681f3Smrg    struct pipe_transfer *transfer;
447ec681f3Smrg    bool is_pipe_secondary;
457ec681f3Smrg    struct nine_subbuffer *buf; /* NULL unless subbuffer are used */
467ec681f3Smrg    bool should_destroy_buf; /* If the subbuffer should be destroyed */
477ec681f3Smrg};
487ec681f3Smrg
497ec681f3Smrgstruct NineBuffer9
507ec681f3Smrg{
517ec681f3Smrg    struct NineResource9 base;
527ec681f3Smrg
537ec681f3Smrg    /* G3D */
547ec681f3Smrg    struct NineTransfer *maps;
557ec681f3Smrg    int nlocks, nmaps, maxmaps;
567ec681f3Smrg    UINT size;
577ec681f3Smrg
587ec681f3Smrg    int16_t bind_count; /* to Device9->state.stream */
597ec681f3Smrg    /* Whether only discard and nooverwrite were used so far
607ec681f3Smrg     * for this buffer. Allows some optimization. */
617ec681f3Smrg    boolean discard_nooverwrite_only;
627ec681f3Smrg    boolean need_sync_if_nooverwrite;
637ec681f3Smrg    struct nine_subbuffer *buf;
647ec681f3Smrg
657ec681f3Smrg    /* Specific to managed buffers */
667ec681f3Smrg    struct {
677ec681f3Smrg        void *data;
687ec681f3Smrg        boolean dirty;
697ec681f3Smrg        struct pipe_box dirty_box; /* region in the resource to update */
707ec681f3Smrg        struct pipe_box upload_pending_regions; /* region with uploads pending */
717ec681f3Smrg        struct list_head list; /* for update_buffers */
727ec681f3Smrg        struct list_head list2; /* for managed_buffers */
737ec681f3Smrg        unsigned pending_upload; /* for uploads */
747ec681f3Smrg        /* SYSTEMMEM DYNAMIC */
757ec681f3Smrg        bool can_unsynchronized; /* Whether the upload can use nooverwrite */
767ec681f3Smrg        struct pipe_box valid_region; /* Region in the GPU buffer with valid content */
777ec681f3Smrg        struct pipe_box required_valid_region; /* Region that needs to be valid right now. */
787ec681f3Smrg        struct pipe_box filled_region; /* Region in the GPU buffer filled since last discard */
797ec681f3Smrg        unsigned num_worker_thread_syncs;
807ec681f3Smrg        unsigned frame_count_last_discard;
817ec681f3Smrg    } managed;
827ec681f3Smrg};
837ec681f3Smrgstatic inline struct NineBuffer9 *
847ec681f3SmrgNineBuffer9( void *data )
857ec681f3Smrg{
867ec681f3Smrg    return (struct NineBuffer9 *)data;
877ec681f3Smrg}
887ec681f3Smrg
897ec681f3SmrgHRESULT
907ec681f3SmrgNineBuffer9_ctor( struct NineBuffer9 *This,
917ec681f3Smrg                        struct NineUnknownParams *pParams,
927ec681f3Smrg                        D3DRESOURCETYPE Type,
937ec681f3Smrg                        DWORD Usage,
947ec681f3Smrg                        UINT Size,
957ec681f3Smrg                        D3DPOOL Pool );
967ec681f3Smrg
977ec681f3Smrgvoid
987ec681f3SmrgNineBuffer9_dtor( struct NineBuffer9 *This );
997ec681f3Smrg
1007ec681f3Smrgstruct pipe_resource *
1017ec681f3SmrgNineBuffer9_GetResource( struct NineBuffer9 *This, unsigned *offset );
1027ec681f3Smrg
1037ec681f3SmrgHRESULT NINE_WINAPI
1047ec681f3SmrgNineBuffer9_Lock( struct NineBuffer9 *This,
1057ec681f3Smrg                        UINT OffsetToLock,
1067ec681f3Smrg                        UINT SizeToLock,
1077ec681f3Smrg                        void **ppbData,
1087ec681f3Smrg                        DWORD Flags );
1097ec681f3Smrg
1107ec681f3SmrgHRESULT NINE_WINAPI
1117ec681f3SmrgNineBuffer9_Unlock( struct NineBuffer9 *This );
1127ec681f3Smrg
1137ec681f3Smrgvoid
1147ec681f3SmrgNineBuffer9_Upload( struct NineBuffer9 *This );
1157ec681f3Smrg
1167ec681f3Smrgstatic void inline
1177ec681f3SmrgNineBindBufferToDevice( struct NineDevice9 *device,
1187ec681f3Smrg                        struct NineBuffer9 **slot,
1197ec681f3Smrg                        struct NineBuffer9 *buf )
1207ec681f3Smrg{
1217ec681f3Smrg    struct NineBuffer9 *old = *slot;
1227ec681f3Smrg
1237ec681f3Smrg    if (buf) {
1247ec681f3Smrg        if ((buf->managed.dirty) && list_is_empty(&buf->managed.list))
1257ec681f3Smrg            list_add(&buf->managed.list, &device->update_buffers);
1267ec681f3Smrg        buf->bind_count++;
1277ec681f3Smrg    }
1287ec681f3Smrg    if (old) {
1297ec681f3Smrg        old->bind_count--;
1307ec681f3Smrg        if (!old->bind_count && old->managed.dirty)
1317ec681f3Smrg            list_delinit(&old->managed.list);
1327ec681f3Smrg    }
1337ec681f3Smrg
1347ec681f3Smrg    nine_bind(slot, buf);
1357ec681f3Smrg}
1367ec681f3Smrg
1377ec681f3Smrgvoid
1387ec681f3SmrgNineBuffer9_SetDirty( struct NineBuffer9 *This );
1397ec681f3Smrg
1407ec681f3Smrg#define BASEBUF_REGISTER_UPDATE(b) { \
1417ec681f3Smrg    if ((b)->managed.dirty && (b)->bind_count) \
1427ec681f3Smrg        if (list_is_empty(&(b)->managed.list)) \
1437ec681f3Smrg            list_add(&(b)->managed.list, &(b)->base.base.device->update_buffers); \
1447ec681f3Smrg    }
1457ec681f3Smrg
1467ec681f3Smrg#endif /* _NINE_BUFFER9_H_ */
147