Attributes:
Your mesh has a number of data streams. For each stream you can keep the following info: (name, type, data).
Upon linking, you can query the GLSL program for active attributes and form an attribute dictionary for this program. Each element here is just (name, type).
When you draw a mesh with a specified GLSL program, you go through programs attribute dictionary and bind the corresponding mesh streams (or reporting an error in case of inconsistency).
Uniforms:
Let the shader parameter dictionary be the set of (name, type, data link). Typically, you can have the following dictionaries:
- Material (diffuse,specular,shininess,etc) – taken from the material
- Engine (camera, model, lights, timers, etc) – taken from engine singleton (global)
- Render (custom parameters related to the shader creator: SSAO radius, blur amount, etc) – provided exclusively by the shader creator class (render)
After linking, the GLSL program is given a set of parameter dictionaries in order to populate it’s own dictionary with the following element format: (location, type, data link). This population is done by querying the list of active uniforms and matching (name, type) pair with the one in dictionaries.
Conclusion:
This method allows for any custom vertex attributes and shader uniforms to be passed, without hard-coded names/semantics in the engine. Basically only the loader and render know about particular semantics:
- Loader fills out the mesh data streams declarations and materials dictionaries.
- Render uses a shader that is aware of the names, provides additional parameters and selects proper meshes to be drawn with.