1 #ifndef __NOUVEAU_STATEOBJ_H__
2 #define __NOUVEAU_STATEOBJ_H__
4 #include "util/u_debug.h"
6 struct nouveau_stateobj_reloc
{
18 struct nouveau_stateobj
{
19 struct pipe_reference reference
;
22 struct nouveau_stateobj_reloc
*reloc
;
29 static INLINE
struct nouveau_stateobj
*
30 so_new(unsigned push
, unsigned reloc
)
32 struct nouveau_stateobj
*so
;
34 so
= MALLOC(sizeof(struct nouveau_stateobj
));
35 pipe_reference_init(&so
->reference
, 1);
36 so
->push
= MALLOC(sizeof(unsigned) * push
);
37 so
->reloc
= MALLOC(sizeof(struct nouveau_stateobj_reloc
) * reloc
);
40 so
->cur_reloc
= so
->cur_packet
= 0;
46 so_ref(struct nouveau_stateobj
*ref
, struct nouveau_stateobj
**pso
)
48 struct nouveau_stateobj
*so
= *pso
;
51 if (pipe_reference(&(*pso
)->reference
, &ref
->reference
)) {
53 for (i
= 0; i
< so
->cur_reloc
; i
++)
54 nouveau_bo_ref(NULL
, &so
->reloc
[i
].bo
);
62 so_data(struct nouveau_stateobj
*so
, unsigned data
)
64 (*so
->cur
++) = (data
);
69 so_datap(struct nouveau_stateobj
*so
, unsigned *data
, unsigned size
)
71 so
->cur_packet
+= (4 * size
);
73 (*so
->cur
++) = (*data
++);
77 so_method(struct nouveau_stateobj
*so
, struct nouveau_grobj
*gr
,
78 unsigned mthd
, unsigned size
)
80 so
->cur_packet
= (gr
->subc
<< 13) | (1 << 18) | (mthd
- 4);
81 so_data(so
, (gr
->subc
<< 13) | (size
<< 18) | mthd
);
85 so_reloc(struct nouveau_stateobj
*so
, struct nouveau_bo
*bo
,
86 unsigned data
, unsigned flags
, unsigned vor
, unsigned tor
)
88 struct nouveau_stateobj_reloc
*r
= &so
->reloc
[so
->cur_reloc
++];
91 nouveau_bo_ref(bo
, &r
->bo
);
92 r
->offset
= so
->cur
- so
->push
;
93 r
->packet
= so
->cur_packet
;
102 so_dump(struct nouveau_stateobj
*so
)
104 unsigned i
, nr
= so
->cur
- so
->push
;
106 for (i
= 0; i
< nr
; i
++)
107 debug_printf("+0x%04x: 0x%08x\n", i
, so
->push
[i
]);
111 so_emit(struct nouveau_channel
*chan
, struct nouveau_stateobj
*so
)
113 struct nouveau_pushbuf
*pb
= chan
->pushbuf
;
116 nr
= so
->cur
- so
->push
;
117 if (pb
->remaining
< nr
)
118 nouveau_pushbuf_flush(chan
, nr
);
121 memcpy(pb
->cur
, so
->push
, nr
* 4);
122 for (i
= 0; i
< so
->cur_reloc
; i
++) {
123 struct nouveau_stateobj_reloc
*r
= &so
->reloc
[i
];
125 nouveau_pushbuf_emit_reloc(chan
, pb
->cur
+ r
->offset
,
126 r
->bo
, r
->data
, 0, r
->flags
,
133 so_emit_reloc_markers(struct nouveau_channel
*chan
, struct nouveau_stateobj
*so
)
135 struct nouveau_pushbuf
*pb
= chan
->pushbuf
;
141 i
= so
->cur_reloc
<< 1;
142 if (pb
->remaining
< i
)
143 nouveau_pushbuf_flush(chan
, i
);
146 for (i
= 0; i
< so
->cur_reloc
; i
++) {
147 struct nouveau_stateobj_reloc
*r
= &so
->reloc
[i
];
149 nouveau_pushbuf_emit_reloc(chan
, pb
->cur
++, r
->bo
, r
->packet
, 0,
150 (r
->flags
& (NOUVEAU_BO_VRAM
|
153 NOUVEAU_BO_DUMMY
, 0, 0);
154 nouveau_pushbuf_emit_reloc(chan
, pb
->cur
++, r
->bo
, r
->data
, 0,
155 r
->flags
| NOUVEAU_BO_DUMMY
,