Skip to content

Commit

Permalink
d3dx9: Support loading mesh user data in D3DXLoadMeshHierarchyFromXIn…
Browse files Browse the repository at this point in the history
…Memory().

CW-Bug-Id: #23002
  • Loading branch information
Paul Gofman committed Nov 21, 2023
1 parent b66053a commit 314da8d
Show file tree
Hide file tree
Showing 2 changed files with 41 additions and 16 deletions.
49 changes: 37 additions & 12 deletions dlls/d3dx9_36/mesh.c
Original file line number Diff line number Diff line change
Expand Up @@ -3830,7 +3830,8 @@ static HRESULT filedata_get_name(ID3DXFileData *filedata, char **name)
}

static HRESULT load_mesh_container(struct ID3DXFileData *filedata, DWORD options, struct IDirect3DDevice9 *device,
struct ID3DXAllocateHierarchy *alloc_hier, D3DXMESHCONTAINER **mesh_container)
struct ID3DXAllocateHierarchy *alloc_hier, D3DXMESHCONTAINER **mesh_container,
struct ID3DXLoadUserData *load_user_data)
{
HRESULT hr;
ID3DXBuffer *adjacency = NULL;
Expand All @@ -3840,6 +3841,10 @@ static HRESULT load_mesh_container(struct ID3DXFileData *filedata, DWORD options
D3DXMESHDATA mesh_data;
DWORD num_materials = 0;
char *name = NULL;
SIZE_T child_count;
ID3DXFileData *child = NULL;
GUID type;
unsigned int i;

mesh_data.Type = D3DXMESHTYPE_MESH;
mesh_data.pMesh = NULL;
Expand All @@ -3852,17 +3857,38 @@ static HRESULT load_mesh_container(struct ID3DXFileData *filedata, DWORD options
hr = filedata_get_name(filedata, &name);
if (FAILED(hr)) goto cleanup;

if (mesh_data.pMesh)
if (!mesh_data.pMesh)
goto cleanup;
hr = alloc_hier->lpVtbl->CreateMeshContainer(alloc_hier, name, &mesh_data,
materials ? ID3DXBuffer_GetBufferPointer(materials) : NULL,
effects ? ID3DXBuffer_GetBufferPointer(effects) : NULL,
num_materials,
adjacency ? ID3DXBuffer_GetBufferPointer(adjacency) : NULL,
skin_info, mesh_container);
if (FAILED(hr) || !load_user_data)
goto cleanup;

hr = filedata->lpVtbl->GetChildren(filedata, &child_count);
if (FAILED(hr))
goto cleanup;

for (i = 0; i < child_count; i++)
{
hr = alloc_hier->lpVtbl->CreateMeshContainer(alloc_hier, name, &mesh_data,
materials ? ID3DXBuffer_GetBufferPointer(materials) : NULL,
effects ? ID3DXBuffer_GetBufferPointer(effects) : NULL,
num_materials,
adjacency ? ID3DXBuffer_GetBufferPointer(adjacency) : NULL,
skin_info, mesh_container);
if (FAILED(hr = filedata->lpVtbl->GetChild(filedata, i, &child)))
goto cleanup;
if (FAILED(hr = child->lpVtbl->GetType(child, &type)))
goto cleanup;

if (!mesh_get_parse_func(&type)
&& FAILED(hr = load_user_data->lpVtbl->LoadMeshChildData(load_user_data, *mesh_container, child)))
goto cleanup;

IUnknown_Release(child);
child = NULL;
}

cleanup:
if (child) IUnknown_Release(child);
if (materials) ID3DXBuffer_Release(materials);
if (effects) ID3DXBuffer_Release(effects);
if (adjacency) ID3DXBuffer_Release(adjacency);
Expand Down Expand Up @@ -3941,7 +3967,7 @@ static HRESULT load_frame(struct ID3DXFileData *filedata, DWORD options, struct
goto err;

if (IsEqualGUID(&type, &TID_D3DRMMesh)) {
hr = load_mesh_container(child, options, device, alloc_hier, next_container);
hr = load_mesh_container(child, options, device, alloc_hier, next_container, load_user_data);
if (SUCCEEDED(hr))
next_container = &(*next_container)->pNextMeshContainer;
} else if (IsEqualGUID(&type, &TID_D3DRMFrameTransformMatrix)) {
Expand Down Expand Up @@ -3987,8 +4013,6 @@ HRESULT WINAPI D3DXLoadMeshHierarchyFromXInMemory(const void *memory, DWORD memo

if (!memory || !memory_size || !device || !frame_hierarchy || !alloc_hier)
return D3DERR_INVALIDCALL;
if (load_user_data)
FIXME("Loading mesh user data not implemented for mesh.\n");

hr = D3DXFileCreate(&d3dxfile);
if (FAILED(hr)) goto cleanup;
Expand Down Expand Up @@ -4022,7 +4046,8 @@ HRESULT WINAPI D3DXLoadMeshHierarchyFromXInMemory(const void *memory, DWORD memo

D3DXMatrixIdentity(&(*next_frame)->TransformationMatrix);

hr = load_mesh_container(filedata, options, device, alloc_hier, &(*next_frame)->pMeshContainer);
hr = load_mesh_container(filedata, options, device, alloc_hier, &(*next_frame)->pMeshContainer,
load_user_data);
if (FAILED(hr)) goto cleanup;
} else if (IsEqualGUID(&guid, &TID_D3DRMFrame)) {
hr = load_frame(filedata, options, device, alloc_hier, next_frame, load_user_data);
Expand Down
8 changes: 4 additions & 4 deletions dlls/d3dx9_36/tests/mesh.c
Original file line number Diff line number Diff line change
Expand Up @@ -2133,20 +2133,20 @@ static void check_user_data(struct test_load_user_data *user_data, unsigned int
{
unsigned int i;

todo_wine ok(user_data->data_count == expected_count, "got %u, expected %u.\n", user_data->data_count, expected_count);
ok(user_data->data_count == expected_count, "got %u, expected %u.\n", user_data->data_count, expected_count);
expected_count = min(expected_count, user_data->data_count);
for (i = 0; i < expected_count; ++i)
{
winetest_push_context("i %u", i);
todo_wine_if(i == 3) ok(user_data->data[i].data_type == expected[i].data_type, "got %u, expected %u.\n",
ok(user_data->data[i].data_type == expected[i].data_type, "got %u, expected %u.\n",
user_data->data[i].data_type, expected[i].data_type);
ok(IsEqualGUID(&user_data->data[i].type, &expected[i].type), "got %s, expected %s.\n",
debugstr_guid(&user_data->data[i].type), debugstr_guid(&expected[i].type));
ok(user_data->data[i].size == expected[i].size, "got %Iu, expected %Iu.\n",
user_data->data[i].size, expected[i].size);
todo_wine_if(i == 3) ok(user_data->data[i].value == expected[i].value, "got %u, expected %u.\n",
ok(user_data->data[i].value == expected[i].value, "got %u, expected %u.\n",
user_data->data[i].value, expected[i].value);
todo_wine_if(i == 3) ok(user_data->data[i].mesh_container == expected[i].mesh_container, "got %u, expected %u.\n",
ok(user_data->data[i].mesh_container == expected[i].mesh_container, "got %u, expected %u.\n",
user_data->data[i].mesh_container, expected[i].mesh_container);
ok(user_data->data[i].num_materials == expected[i].num_materials, "got %u, expected %u.\n",
user_data->data[i].num_materials, expected[i].num_materials);
Expand Down

0 comments on commit 314da8d

Please sign in to comment.