d3d10: Parse effect shaders as anonymous shaders.

This commit is contained in:
Rico Schüller 2010-01-10 19:30:42 +01:00 committed by Alexandre Julliard
parent 0682453d14
commit 0b4e6c71b5
2 changed files with 189 additions and 71 deletions

View File

@ -45,6 +45,12 @@ enum d3d10_effect_object_type
D3D10_EOT_GEOMETRYSHADER = 8, D3D10_EOT_GEOMETRYSHADER = 8,
}; };
enum d3d10_effect_object_operation
{
D3D10_EOO_VALUE = 1,
D3D10_EOO_ANONYMOUS_SHADER = 7,
};
struct d3d10_effect_object struct d3d10_effect_object
{ {
struct d3d10_effect_pass *pass; struct d3d10_effect_pass *pass;
@ -105,6 +111,7 @@ struct d3d10_effect_variable
struct d3d10_effect_type *type; struct d3d10_effect_type *type;
struct d3d10_effect *effect; struct d3d10_effect *effect;
void *data;
char *name; char *name;
char *semantic; char *semantic;
DWORD buffer_offset; DWORD buffer_offset;
@ -143,6 +150,12 @@ struct d3d10_effect_technique
struct d3d10_effect_variable *annotations; struct d3d10_effect_variable *annotations;
}; };
struct d3d10_effect_anonymous_shader
{
struct d3d10_effect_variable shader;
struct d3d10_effect_type type;
};
/* ID3D10Effect */ /* ID3D10Effect */
extern const struct ID3D10EffectVtbl d3d10_effect_vtbl DECLSPEC_HIDDEN; extern const struct ID3D10EffectVtbl d3d10_effect_vtbl DECLSPEC_HIDDEN;
struct d3d10_effect struct d3d10_effect
@ -167,11 +180,14 @@ struct d3d10_effect
DWORD rendertargetview_count; DWORD rendertargetview_count;
DWORD depthstencilview_count; DWORD depthstencilview_count;
DWORD shader_call_count; DWORD shader_call_count;
DWORD shader_compile_count; DWORD anonymous_shader_count;
DWORD anonymous_shader_current;
struct wine_rb_tree types; struct wine_rb_tree types;
struct d3d10_effect_variable *local_buffers; struct d3d10_effect_variable *local_buffers;
struct d3d10_effect_variable *local_variables; struct d3d10_effect_variable *local_variables;
struct d3d10_effect_anonymous_shader *anonymous_shaders;
struct d3d10_effect_technique *techniques; struct d3d10_effect_technique *techniques;
}; };

View File

@ -266,43 +266,49 @@ static HRESULT shader_chunk_handler(const char *data, DWORD data_size, DWORD tag
return S_OK; return S_OK;
} }
static HRESULT parse_shader(struct d3d10_effect_object *o, const char *data) static HRESULT parse_shader(struct d3d10_effect_variable *v, const char *data)
{ {
ID3D10Device *device = o->pass->technique->effect->device; ID3D10Device *device = v->effect->device;
struct d3d10_effect_shader_variable *s; struct d3d10_effect_shader_variable *s;
const char *ptr = data; const char *ptr = data;
DWORD dxbc_size; DWORD dxbc_size;
HRESULT hr; HRESULT hr;
o->data = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(struct d3d10_effect_shader_variable)); s = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*s));
if (!o->data) if (!s)
{ {
ERR("Failed to allocate shader variable memory\n"); ERR("Failed to allocate shader variable memory\n");
return E_OUTOFMEMORY; return E_OUTOFMEMORY;
} }
if (!ptr) return S_OK; v->data = s;
s = o->data; if (!ptr) return S_OK;
read_dword(&ptr, &dxbc_size); read_dword(&ptr, &dxbc_size);
TRACE("dxbc size: %#x\n", dxbc_size); TRACE("dxbc size: %#x\n", dxbc_size);
switch (o->type) switch (v->type->basetype)
{ {
case D3D10_EOT_VERTEXSHADER: case D3D10_SVT_VERTEXSHADER:
hr = ID3D10Device_CreateVertexShader(device, ptr, dxbc_size, &s->shader.vs); hr = ID3D10Device_CreateVertexShader(device, ptr, dxbc_size, &s->shader.vs);
if (FAILED(hr)) return hr; if (FAILED(hr)) return hr;
break; break;
case D3D10_EOT_PIXELSHADER: case D3D10_SVT_PIXELSHADER:
hr = ID3D10Device_CreatePixelShader(device, ptr, dxbc_size, &s->shader.ps); hr = ID3D10Device_CreatePixelShader(device, ptr, dxbc_size, &s->shader.ps);
if (FAILED(hr)) return hr; if (FAILED(hr)) return hr;
break; break;
case D3D10_EOT_GEOMETRYSHADER:
case D3D10_SVT_GEOMETRYSHADER:
hr = ID3D10Device_CreateGeometryShader(device, ptr, dxbc_size, &s->shader.gs); hr = ID3D10Device_CreateGeometryShader(device, ptr, dxbc_size, &s->shader.gs);
if (FAILED(hr)) return hr; if (FAILED(hr)) return hr;
break; break;
default:
ERR("This should not happen!\n");
return E_FAIL;
break;
} }
return parse_dxbc(ptr, dxbc_size, shader_chunk_handler, s); return parse_dxbc(ptr, dxbc_size, shader_chunk_handler, s);
@ -820,56 +826,140 @@ static HRESULT parse_fx10_annotation(struct d3d10_effect_variable *a, const char
return S_OK; return S_OK;
} }
static HRESULT parse_fx10_anonymous_shader(struct d3d10_effect *e, struct d3d10_effect_anonymous_shader *s,
enum d3d10_effect_object_type otype)
{
struct d3d10_effect_variable *v = &s->shader;
struct d3d10_effect_type *t = &s->type;
const char *shader = NULL;
switch (otype)
{
case D3D10_EOT_VERTEXSHADER:
shader = "vertexshader";
t->basetype = D3D10_SVT_VERTEXSHADER;
break;
case D3D10_EOT_PIXELSHADER:
shader = "pixelshader";
t->basetype = D3D10_SVT_PIXELSHADER;
break;
case D3D10_EOT_GEOMETRYSHADER:
shader = "geometryshader";
t->basetype = D3D10_SVT_GEOMETRYSHADER;
break;
}
if (!copy_name(shader, &t->name))
{
ERR("Failed to copy name.\n");
return E_OUTOFMEMORY;
}
TRACE("Type name: %s.\n", debugstr_a(t->name));
t->type_class = D3D10_SVC_OBJECT;
t->vtbl = &d3d10_effect_type_vtbl;
v->type = t;
v->effect = e;
set_variable_vtbl(v);
if (!copy_name("$Anonymous", &v->name))
{
ERR("Failed to copy semantic.\n");
return E_OUTOFMEMORY;
}
TRACE("Variable name: %s.\n", debugstr_a(v->name));
if (!copy_name(NULL, &v->semantic))
{
ERR("Failed to copy semantic.\n");
return E_OUTOFMEMORY;
}
TRACE("Variable semantic: %s.\n", debugstr_a(v->semantic));
return S_OK;
}
static HRESULT parse_fx10_object(struct d3d10_effect_object *o, const char **ptr, const char *data) static HRESULT parse_fx10_object(struct d3d10_effect_object *o, const char **ptr, const char *data)
{ {
const char *data_ptr; const char *data_ptr = NULL;
DWORD offset; DWORD offset, index;
enum d3d10_effect_object_operation operation;
HRESULT hr; HRESULT hr;
struct d3d10_effect *effect = o->pass->technique->effect;
read_dword(ptr, &o->type); read_dword(ptr, &o->type);
TRACE("Effect object is of type %#x.\n", o->type); TRACE("Effect object is of type %#x.\n", o->type);
skip_dword_unknown(ptr, 2); read_dword(ptr, &index);
TRACE("Effect object index %#x.\n", index);
read_dword(ptr, &operation);
TRACE("Effect object operation %#x.\n", operation);
read_dword(ptr, &offset); read_dword(ptr, &offset);
TRACE("Effect object idx is at offset %#x.\n", offset); TRACE("Effect object idx is at offset %#x.\n", offset);
data_ptr = data + offset; switch(operation)
read_dword(&data_ptr, &offset);
TRACE("Effect object starts at offset %#x.\n", offset);
/* FIXME: This probably isn't completely correct. */
if (offset == 1)
{ {
WARN("Skipping effect object.\n"); /* FIXME: This probably isn't completely correct. */
data_ptr = NULL; case D3D10_EOO_VALUE:
} FIXME("Copy variable value\n");
else hr = E_FAIL;
{
data_ptr = data + offset;
}
switch (o->type)
{
case D3D10_EOT_VERTEXSHADER:
TRACE("Vertex shader\n");
hr = parse_shader(o, data_ptr);
break; break;
case D3D10_EOT_PIXELSHADER: case D3D10_EOO_ANONYMOUS_SHADER:
TRACE("Pixel shader\n"); TRACE("Anonymous shader\n");
hr = parse_shader(o, data_ptr);
break;
case D3D10_EOT_GEOMETRYSHADER: /* check anonymous_shader_current for validity */
TRACE("Geometry shader\n"); if (effect->anonymous_shader_current >= effect->anonymous_shader_count)
hr = parse_shader(o, data_ptr); {
ERR("Anonymous shader count is wrong!\n");
return E_FAIL;
}
data_ptr = data + offset;
read_dword(&data_ptr, &offset);
TRACE("Effect object starts at offset %#x.\n", offset);
data_ptr = data + offset;
hr = parse_fx10_anonymous_shader(effect, &effect->anonymous_shaders[effect->anonymous_shader_current], o->type);
if (FAILED(hr)) return hr;
o->data = &effect->anonymous_shaders[effect->anonymous_shader_current].shader;
++effect->anonymous_shader_current;
switch (o->type)
{
case D3D10_EOT_VERTEXSHADER:
TRACE("Vertex shader\n");
hr = parse_shader(o->data, data_ptr);
break;
case D3D10_EOT_PIXELSHADER:
TRACE("Pixel shader\n");
hr = parse_shader(o->data, data_ptr);
break;
case D3D10_EOT_GEOMETRYSHADER:
TRACE("Geometry shader\n");
hr = parse_shader(o->data, data_ptr);
break;
default:
FIXME("Unhandled object type %#x\n", o->type);
hr = E_FAIL;
break;
}
break; break;
default: default:
FIXME("Unhandled object type %#x\n", o->type);
hr = E_FAIL; hr = E_FAIL;
FIXME("Unhandled operation %#x.\n", operation);
break; break;
} }
@ -1408,6 +1498,13 @@ static HRESULT parse_fx10_body(struct d3d10_effect *e, const char *data, DWORD d
return E_OUTOFMEMORY; return E_OUTOFMEMORY;
} }
e->anonymous_shaders = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, e->anonymous_shader_count * sizeof(*e->anonymous_shaders));
if (!e->anonymous_shaders)
{
ERR("Failed to allocate techniques memory\n");
return E_OUTOFMEMORY;
}
e->techniques = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, e->technique_count * sizeof(*e->techniques)); e->techniques = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, e->technique_count * sizeof(*e->techniques));
if (!e->techniques) if (!e->techniques)
{ {
@ -1513,8 +1610,8 @@ static HRESULT parse_fx10(struct d3d10_effect *e, const char *data, DWORD data_s
read_dword(&ptr, &e->shader_call_count); read_dword(&ptr, &e->shader_call_count);
TRACE("Shader call count: %u\n", e->shader_call_count); TRACE("Shader call count: %u\n", e->shader_call_count);
read_dword(&ptr, &e->shader_compile_count); read_dword(&ptr, &e->anonymous_shader_count);
TRACE("Shader compile count: %u\n", e->shader_compile_count); TRACE("Anonymous shader count: %u\n", e->anonymous_shader_count);
return parse_fx10_body(e, ptr, data_size - (ptr - data)); return parse_fx10_body(e, ptr, data_size - (ptr - data));
} }
@ -1543,42 +1640,25 @@ HRESULT d3d10_effect_parse(struct d3d10_effect *This, const void *data, SIZE_T d
return parse_dxbc(data, data_size, fx10_chunk_handler, This); return parse_dxbc(data, data_size, fx10_chunk_handler, This);
} }
static void d3d10_effect_object_destroy(struct d3d10_effect_object *o)
{
TRACE("effect object %p.\n", o);
switch(o->type)
{
case D3D10_EOT_VERTEXSHADER:
case D3D10_EOT_PIXELSHADER:
case D3D10_EOT_GEOMETRYSHADER:
HeapFree(GetProcessHeap(), 0, ((struct d3d10_effect_shader_variable *)o->data)->input_signature);
break;
default:
break;
}
HeapFree(GetProcessHeap(), 0, o->data);
}
static HRESULT d3d10_effect_object_apply(struct d3d10_effect_object *o) static HRESULT d3d10_effect_object_apply(struct d3d10_effect_object *o)
{ {
ID3D10Device *device = o->pass->technique->effect->device; ID3D10Device *device = o->pass->technique->effect->device;
struct d3d10_effect_variable *v = (struct d3d10_effect_variable*) o->data;
TRACE("effect object %p, type %#x.\n", o, o->type); TRACE("effect object %p, type %#x.\n", o, o->type);
switch(o->type) switch(o->type)
{ {
case D3D10_EOT_VERTEXSHADER: case D3D10_EOT_VERTEXSHADER:
ID3D10Device_VSSetShader(device, ((struct d3d10_effect_shader_variable *)o->data)->shader.vs); ID3D10Device_VSSetShader(device, ((struct d3d10_effect_shader_variable *)v->data)->shader.vs);
return S_OK; return S_OK;
case D3D10_EOT_PIXELSHADER: case D3D10_EOT_PIXELSHADER:
ID3D10Device_PSSetShader(device, ((struct d3d10_effect_shader_variable *)o->data)->shader.ps); ID3D10Device_PSSetShader(device, ((struct d3d10_effect_shader_variable *)v->data)->shader.ps);
return S_OK; return S_OK;
case D3D10_EOT_GEOMETRYSHADER: case D3D10_EOT_GEOMETRYSHADER:
ID3D10Device_GSSetShader(device, ((struct d3d10_effect_shader_variable *)o->data)->shader.gs); ID3D10Device_GSSetShader(device, ((struct d3d10_effect_shader_variable *)v->data)->shader.gs);
return S_OK; return S_OK;
default: default:
@ -1621,6 +1701,22 @@ static void d3d10_effect_variable_destroy(struct d3d10_effect_variable *v)
} }
HeapFree(GetProcessHeap(), 0, v->elements); HeapFree(GetProcessHeap(), 0, v->elements);
} }
if (v->data)
{
switch(v->type->basetype)
{
case D3D10_SVT_VERTEXSHADER:
case D3D10_SVT_PIXELSHADER:
case D3D10_SVT_GEOMETRYSHADER:
HeapFree(GetProcessHeap(), 0, ((struct d3d10_effect_shader_variable *)v->data)->input_signature);
break;
default:
break;
}
HeapFree(GetProcessHeap(), 0, v->data);
}
} }
static void d3d10_effect_pass_destroy(struct d3d10_effect_pass *p) static void d3d10_effect_pass_destroy(struct d3d10_effect_pass *p)
@ -1632,10 +1728,6 @@ static void d3d10_effect_pass_destroy(struct d3d10_effect_pass *p)
HeapFree(GetProcessHeap(), 0, p->name); HeapFree(GetProcessHeap(), 0, p->name);
if (p->objects) if (p->objects)
{ {
for (i = 0; i < p->object_count; ++i)
{
d3d10_effect_object_destroy(&p->objects[i]);
}
HeapFree(GetProcessHeap(), 0, p->objects); HeapFree(GetProcessHeap(), 0, p->objects);
} }
@ -1647,7 +1739,6 @@ static void d3d10_effect_pass_destroy(struct d3d10_effect_pass *p)
} }
HeapFree(GetProcessHeap(), 0, p->annotations); HeapFree(GetProcessHeap(), 0, p->annotations);
} }
} }
static void d3d10_effect_technique_destroy(struct d3d10_effect_technique *t) static void d3d10_effect_technique_destroy(struct d3d10_effect_technique *t)
@ -1783,6 +1874,16 @@ static ULONG STDMETHODCALLTYPE d3d10_effect_Release(ID3D10Effect *iface)
HeapFree(GetProcessHeap(), 0, This->local_buffers); HeapFree(GetProcessHeap(), 0, This->local_buffers);
} }
if (This->anonymous_shaders)
{
for (i = 0; i < This->anonymous_shader_count; ++i)
{
d3d10_effect_variable_destroy(&This->anonymous_shaders[i].shader);
HeapFree(GetProcessHeap(), 0, &This->anonymous_shaders[i].type.name);
}
HeapFree(GetProcessHeap(), 0, This->anonymous_shaders);
}
wine_rb_destroy(&This->types, d3d10_effect_type_destroy, NULL); wine_rb_destroy(&This->types, d3d10_effect_type_destroy, NULL);
ID3D10Device_Release(This->device); ID3D10Device_Release(This->device);
@ -2212,7 +2313,8 @@ static HRESULT STDMETHODCALLTYPE d3d10_effect_pass_GetDesc(ID3D10EffectPass *ifa
struct d3d10_effect_object *o = &This->objects[i]; struct d3d10_effect_object *o = &This->objects[i];
if (o->type == D3D10_EOT_VERTEXSHADER) if (o->type == D3D10_EOT_VERTEXSHADER)
{ {
struct d3d10_effect_shader_variable *s = o->data; struct d3d10_effect_variable *v = o->data;
struct d3d10_effect_shader_variable *s = v->data;
desc->pIAInputSignature = (BYTE *)s->input_signature; desc->pIAInputSignature = (BYTE *)s->input_signature;
desc->IAInputSignatureSize = s->input_signature_size; desc->IAInputSignatureSize = s->input_signature_size;
break; break;