1 #include "pipe/p_context.h"
3 #include "nv40_context.h"
6 struct nouveau_resource
*object
;
12 static INLINE
struct nv40_query
*
13 nv40_query(struct pipe_query
*pipe
)
15 return (struct nv40_query
*)pipe
;
18 static struct pipe_query
*
19 nv40_query_create(struct pipe_context
*pipe
, unsigned query_type
)
23 q
= CALLOC(1, sizeof(struct nv40_query
));
26 return (struct pipe_query
*)q
;
30 nv40_query_destroy(struct pipe_context
*pipe
, struct pipe_query
*pq
)
32 struct nv40_query
*q
= nv40_query(pq
);
35 nouveau_resource_free(&q
->object
);
40 nv40_query_begin(struct pipe_context
*pipe
, struct pipe_query
*pq
)
42 struct nv40_context
*nv40
= nv40_context(pipe
);
43 struct nv40_query
*q
= nv40_query(pq
);
44 struct nv40_screen
*screen
= nv40
->screen
;
45 struct nouveau_channel
*chan
= screen
->base
.channel
;
46 struct nouveau_grobj
*curie
= screen
->curie
;
48 assert(q
->type
== PIPE_QUERY_OCCLUSION_COUNTER
);
50 /* Happens when end_query() is called, then another begin_query()
51 * without querying the result in-between. For now we'll wait for
52 * the existing query to notify completion, but it could be better.
56 pipe
->get_query_result(pipe
, pq
, 1, &tmp
);
59 if (nouveau_resource_alloc(nv40
->screen
->query_heap
, 1, NULL
, &q
->object
))
61 nouveau_notifier_reset(nv40
->screen
->query
, q
->object
->start
);
63 BEGIN_RING(chan
, curie
, NV40TCL_QUERY_RESET
, 1);
65 BEGIN_RING(chan
, curie
, NV40TCL_QUERY_UNK17CC
, 1);
72 nv40_query_end(struct pipe_context
*pipe
, struct pipe_query
*pq
)
74 struct nv40_context
*nv40
= nv40_context(pipe
);
75 struct nv40_query
*q
= nv40_query(pq
);
76 struct nv40_screen
*screen
= nv40
->screen
;
77 struct nouveau_channel
*chan
= screen
->base
.channel
;
78 struct nouveau_grobj
*curie
= screen
->curie
;
80 BEGIN_RING(chan
, curie
, NV40TCL_QUERY_GET
, 1);
81 OUT_RING (chan
, (0x01 << NV40TCL_QUERY_GET_UNK24_SHIFT
) |
82 ((q
->object
->start
* 32) << NV40TCL_QUERY_GET_OFFSET_SHIFT
));
87 nv40_query_result(struct pipe_context
*pipe
, struct pipe_query
*pq
,
88 boolean wait
, uint64_t *result
)
90 struct nv40_context
*nv40
= nv40_context(pipe
);
91 struct nv40_query
*q
= nv40_query(pq
);
93 assert(q
->object
&& q
->type
== PIPE_QUERY_OCCLUSION_COUNTER
);
98 status
= nouveau_notifier_status(nv40
->screen
->query
,
100 if (status
!= NV_NOTIFY_STATE_STATUS_COMPLETED
) {
103 nouveau_notifier_wait_status(nv40
->screen
->query
,
105 NV_NOTIFY_STATE_STATUS_COMPLETED
,
109 q
->result
= nouveau_notifier_return_val(nv40
->screen
->query
,
112 nouveau_resource_free(&q
->object
);
120 nv40_init_query_functions(struct nv40_context
*nv40
)
122 nv40
->pipe
.create_query
= nv40_query_create
;
123 nv40
->pipe
.destroy_query
= nv40_query_destroy
;
124 nv40
->pipe
.begin_query
= nv40_query_begin
;
125 nv40
->pipe
.end_query
= nv40_query_end
;
126 nv40
->pipe
.get_query_result
= nv40_query_result
;