-
Notifications
You must be signed in to change notification settings - Fork 71
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[FAQ] Common mistakes when parsing SMPL UV data #4
Comments
How did you extract the UV map? From the FBX file? In the code it looks like you're parsing a file named 'SMPL_template_UV_map.obj'. Do you have that file in the repository? |
@radvani I downloaded the FBX file from official website and convert it into .obj using an online converter , the result is the aforementioned |
Interesting, today I'm trying to extract the UV map directly from the FBX file, using the FBX SDK. I will let you know if it looks different than yours (perhaps the FBX -> OBJ conversion corrupted something?) |
Cool! You can visualize the results with the following code snippets:
|
Which FBX file were you using from the official website? Was it basicModel_f_lbs_10_207_0_v1.0.2, or basicModel_m_lbs_10_207_0_v1.0.2? My preliminary printout of texcoords from FBX show that they're different from what's in your OBJ file. Also, regarding visualizing the results, I think I'm missing something. It looks like you're generating the human model (shown above) by creating the PLY file using the snippet in the previous comment, but how are you generating the images of the red/blue UV map? |
Nevermind my first question above, it looks like you're using the female model. I generated a new OBJ and UV map directly from the FBX file and will try to test it now (still working through how to do that exactly). |
Weird. As I recall SMPL model is gender-independent, and I never bother to find the difference... I guess male model has an extra "part" eh? |
Actually it may take me a bit to test this myself, but I've attached the OBJ I generated from the FBX SDK in case you want to try out. You can use your usual methods to generate the UV map from this (in get_SMPL_UV_map.py). |
I'm getting the following when using your OBJ file, and it seems fairly consistent? I wonder if there's actually an issue in get_SMPL_UV_map, in that in your version you're creating a 1-1 mapping between texcoord indices and vertices. I don't think that's correct, in that a single texcoord index can correspond to multiple vertices (there are no restrictions against this in OBJ). I changed this so that instead there's a mapping between faces: e.g. each tuple of texcoords maps to the corresponding tuple of vertices. This resulted in my graphic above. |
I think you got a point, it's indeed much safer to use face correspondences for UV plot... Think you can submit a PR? |
Sure, I think my changes broke the render method in that class but I'll fix that and submit a PR tonight. |
@radvani You are absolutely right that texcoords and vertices are not in one-to-one correspondance. But what's puzzling me is that even the same texture vertex could also correspond to multiple 3D vertices, as shown here: There are five such points in total: That's weird as hell, it seems that the UV texture generation process does not just cut mesh into pieces, but also attach some pieces together at these weird points... I wonder what do you make of this? |
That's interesting, my first guess it that the online OBJ converter you're using is trying to minimize the number of unique texture coordinates using too large of an epsilon, causing it to merge together texture coordinates that are in fact distinct. The reason this can happen is that the FBX format actually doesn't even define a mapping between texture coordinates and vertices. It just defines:
(Unless I'm missing something, but I've spent awhile staring at this cryptic FBX SDK) For our model, that means there are 6890 vertices defined in the FBX file, and 41319 texture coordinates. This results in a massive amount of duplicated texture coordinates, which is why most converters minimize them. I can try to do my own OBJ export with smarter texture coordinate minimization, but I realized I don't have the gender neutral FBX file that corresponds to neutral_smpl_with_cocoplus_reg.pkl. I only have the male and female versions. Do you have the FBX from which you generated the OBJ? |
Sure, you can find the FBX file via this link, it contains three animations, I used this one: Also, here's some of the test result I generated from h36m dataset: Google link It contains multiple things, including some real 3D meshes sampled and aligned with the corresponding 2D image. Can you test your Now I'm not sure if that online converter I used is faulty, or the .fbx file is corrupted in the first place. Anyway, I'll try to use FBX-SDK for generation, and try to use the other two fbx files for UV data extraction. I'll keep you informed about any new findings. |
That sounds good. Actually the OBJ I pasted above (converted from FBX) which didn't do any texture coordinate minimization (and therefore had 41319 texture coordinates), looks quite a bit different from yours. Most of this is probably due to the fact that I used per-corner normals instead of per-vertex, but it goes to show how the conversion can really make an impact. |
Haha you're right they're quite the same on the surface, it's just the normals and (probably) the texture coordinates that are different. |
@radvani I think we can draw the conclusion on this issue, but I still feel safer if you can double check my findings below: Today I converted the fbx file into .ply format, which generates completely different results: Some of the original 6890 vertices are duplicated, making a total of 8964, and formed an exact 1-1 correspondance with UV vertices. Now every uv vertex is assigned to a unique 3d vertex, and the color of each uv vertex is uniquely determined without any ambiguity. However, the generated UV maps still suffers from the color inconsistency, like this: By now I'm pretty certain that the real problem lies in the vt_faces representation: some edges in the uv domain actually connects two uv vertices that are far away on real 3D mesh. The only uncertainty remains, is whether the original fbx file contains the wrong vt_face info in the first place, or some error happens during the process of format conversion? To verify that, I've also tested my algorithm with your attached obj file that you converted from FBX-SDK, and the error remains the same. I feel it safe to say that the official SMPL data is corrupted in the first place. |
Interesting! Yes, I'll take a look on Monday and see if my results are similar to yours. |
@radvani Hey, I managed to retrieve some information from the .fbx file with both FBX C++ SDK and Python SDK, but the results seem weird: For python results For C++ Results The result is also very weird, the Just wonder what kind SDK did you use to parse the following obj file? Is it possible to share me the conversion code so I can cross check it? Thanks:)
|
Hey! I noticed the same thing regarding the use of quads instead of triangles, so I run it through the FBX SDK's triangulator. You can do this like so:
Where the scene is your root FbxScene, and fbxManager is your FBXManager. This is using the C++ FBX SDK. I can send over my conversion code soon I'm just cleaning it up a bit. I'm going to check on my vertex count now... |
Pre-triangulation, I'm getting 6890 vertices, and 6900 polygons (quads). Post-triangulation, I get 6890 vertices and 13773 polygons (triangles). |
Attached is my C++ FBX to OBJ converter. It's meant to run on a Mac but you can get it to run on Linux pretty easily by just changing the log function it's using at the top of the implementation file. Most of the magic happens in VROFbxToObjConverter::convertToObj. Before that it's just a lot of setup. I ripped this out of a library I have that does more generalized FBX export (e.g. skeletal animations, etc.) so there may be some things in there that aren't strictly necessary. To use, just create a VROFbxToObjConverter and call convertToObj with source path and destination path, like this:
Note that this converter does no collapsing of duplicate texture coordinates, so it ends up with a ton of them. The only dependencies are STL and the FBX SDK. |
Ok I think I see what you mean by the color maps being off. When I render just the X values to the scatter plot -- encoding X in the 'R' channel of the UV map, and leaving G and B at zero -- I would expect red to increase from left to right (for my model, where arms are outstretched and the character is facing the camera), but it doesn't. See first image. However, when I do this same thing using just the mesh's vertices (as opposed to the positions in the 2D image), it works better, but maybe still not perfect (second image). Looking into it now. |
Update: the issue appears to be that the SMPL models we generate at runtime (through smpl_torch_generator.py) do not match the template SMPL from which we derived the UV coordinates. More specifically, we rely on the correspondence from the texcoord indices to the vertex coord indices to be identical in both the template SMPL and the runtime SMPL. We require this so that we can iterate through each face in the template SMPL and find the texture coordinates in the template SMPL and the corresponding vertices in the runtime SMPL. We then write this correspondence to the UV position map. This is why the scatter plot of the template OBJ's own vertices looks good, but as soon as we pass in another OBJ, we have discontinuities. |
Ok, so I had a 3D modeling friend create a new 2D UV map for us. This is not the same as the template SMPL UV map (the parts are arranged in different places) but it does maintain the correspondence between the runtime SMPL and the template SMPL. The results on running it from a generated SMPL model are shown here (ignore the fact that my model is poorly aligned, that's a separate bug). |
To use this one, use get_SMPL_UV_map on the OBJ I attached here, then run create_UV_position_maps as normal... just make sure you delete the barycentric coordinate cache file first! |
That’s very inspiring news! I should have thought about it when I first notice the difference between UV maps from SMPL templates and generated meshes. It’s really nice to have a working UV maps though. I will merge this into the current framework ASAP. This could be the game changer for our projects:) |
@radvani I just finished baseline training with some interesting results. I've sent you an email at |
@Lotayou Good job. |
We actually tried that, but the UV map extracted from the FBX templates doesn't share the same vertex to texture coordinate mapping as the meshes produced by the SMPL generator at runtime (see comments above). Is there a way you know of to make that work? |
@radvani I tried your code. I see that you use FbxGeometryConverter to triangulate the mesh which I didn't use. Instead I get 6900 quads and save each quad into 2 triangles manually. The index is got by GetPolygonVertex. |
@ypfill This result actually looks promising. However I wonder if it will work on synthesized 3D mesh too, besides the SMPL template. Do you wanna share a converted .obj file or a processing script so I can acquire the aforementioned triangulation result? You can either attach it here or send me an email via |
@Lotayou Sorry that I can't share code or data for some reason. Any discussion is welcomed. |
@ypflll Hi, your job seems good! I wonder which coordinate system of body vertices is used to be as the uv position map's pixel rgb value. Cause i use coordinate system of image and get the same uv position map as radvani get. Maybe you use the world coordinate system, i guess? |
@radvani Hi, I have get the same uv position map as your firstly got, and i use it to get almost right coordinates of body in image. But the uv position map does seems weird. |
@Dorniwang We use coordinates aligned in the image plane. |
@ypflll It's all right. It just that something doesn't add up according to your descriptions:
BTW, did you use my script for UV map generation or you write it on your own? |
@Lotayou Hi, some quads (about 24 ) in SMPL only has three vertices, so the total number of triangles after split is just 13776. But I has the same question about how to split it, because I had done this half month ago, but i did not get the right triangles, and the index of triangles is only for control point, which is model vertices, and has no enough information about map between vertex and uv. |
@Lotayou |
@ypflll @radvani @Dorniwang Good news guys, I've successfully parsed SMPL original UV maps! The trick is to use Blender software for fbx2obj conversion, which is way easier than that cryptic FBX-SDK. After that, run |
Nice job! I tried it out and it worked on my end. |
@Lotayou Amazing! I'll try it! |
@Lotayou Usinig the new uv map, I have got the right uv position map! That's greate! |
Hi, when I run data_utils/triangulation.py, it meets "KeyError: 'joint_regressor'" |
@Lotayou (a.k.a myself) I just found that my SMPL UV data at hand is messed up, UV vertices' topology does not match that of SMPL 3D mesh. Here's the result:
This error hampers my further development since the color inconsistency induced by UV data error serious affects the interpolation accuracy of (x,y,z) coords. I'm afraid I cannot do anything further until this problem is solved.
Any help would be deeply appreciated!
The text was updated successfully, but these errors were encountered: