goat3d
changeset 52:cb5414f406eb
merged
author | John Tsiombikas <nuclear@member.fsf.org> |
---|---|
date | Fri, 17 Jan 2014 18:27:47 +0200 (2014-01-17) |
parents | fa5c52ea9d59 0be413ac2e0a |
children | 6d514398a728 |
files | src/goat3d.cc src/goat3d_readxml.cc src/goat3d_write.cc |
diffstat | 10 files changed, 729 insertions(+), 110 deletions(-) [+] |
line diff
1.1 --- a/Makefile Fri Jan 17 18:16:09 2014 +0200 1.2 +++ b/Makefile Fri Jan 17 18:27:47 2014 +0200 1.3 @@ -37,7 +37,7 @@ 1.4 CC = clang 1.5 CXX = clang++ 1.6 CXXFLAGS = -pedantic -Wall $(dbg) $(opt) $(pic) $(extinc) 1.7 -LDFLAGS = $(extlibs) 1.8 +LDFLAGS = $(extlibs) -lpthread 1.9 1.10 .PHONY: all 1.11 all: $(lib_so) $(lib_a) 1.12 @@ -48,15 +48,19 @@ 1.13 $(lib_a): $(obj) $(extlibs) 1.14 $(AR) rcs $@ $(obj) $(extlibs) 1.15 1.16 +.PHONY: $(openctm) 1.17 $(openctm): 1.18 $(MAKE) -C libs/openctm 1.19 1.20 +.PHONY: $(tinyxml2) 1.21 $(tinyxml2): 1.22 $(MAKE) -C libs/tinyxml2 1.23 1.24 +.PHONY: $(vmath) 1.25 $(vmath): 1.26 $(MAKE) -C libs/vmath 1.27 1.28 +.PHONY: $(anim) 1.29 $(anim): 1.30 $(MAKE) -C libs/anim 1.31
2.1 --- a/libs/anim/anim.c Fri Jan 17 18:16:09 2014 +0200 2.2 +++ b/libs/anim/anim.c Fri Jan 17 18:27:47 2014 +0200 2.3 @@ -8,7 +8,7 @@ 2.4 2.5 static void invalidate_cache(struct anm_node *node); 2.6 2.7 -int anm_init_node(struct anm_node *node) 2.8 +int anm_init_animation(struct anm_animation *anim) 2.9 { 2.10 int i, j; 2.11 static const float defaults[] = { 2.12 @@ -17,19 +17,59 @@ 2.13 1.0f, 1.0f, 1.0f /* default scale factor */ 2.14 }; 2.15 2.16 + anim->name = 0; 2.17 + 2.18 + for(i=0; i<ANM_NUM_TRACKS; i++) { 2.19 + if(anm_init_track(anim->tracks + i) == -1) { 2.20 + for(j=0; j<i; j++) { 2.21 + anm_destroy_track(anim->tracks + i); 2.22 + } 2.23 + } 2.24 + anm_set_track_default(anim->tracks + i, defaults[i]); 2.25 + } 2.26 + return 0; 2.27 +} 2.28 + 2.29 +void anm_destroy_animation(struct anm_animation *anim) 2.30 +{ 2.31 + int i; 2.32 + for(i=0; i<ANM_NUM_TRACKS; i++) { 2.33 + anm_destroy_track(anim->tracks + i); 2.34 + } 2.35 + free(anim->name); 2.36 +} 2.37 + 2.38 +void anm_set_animation_name(struct anm_animation *anim, const char *name) 2.39 +{ 2.40 + char *newname = malloc(strlen(name) + 1); 2.41 + if(!newname) return; 2.42 + 2.43 + strcpy(newname, name); 2.44 + 2.45 + free(anim->name); 2.46 + anim->name = newname; 2.47 +} 2.48 + 2.49 +/* ---- node implementation ----- */ 2.50 + 2.51 +int anm_init_node(struct anm_node *node) 2.52 +{ 2.53 memset(node, 0, sizeof *node); 2.54 2.55 - for(i=0; i<ANM_NUM_TRACKS; i++) { 2.56 - if(anm_init_track(node->tracks + i) == -1) { 2.57 - for(j=0; j<i; j++) { 2.58 - anm_destroy_track(node->tracks + i); 2.59 - } 2.60 - } 2.61 - anm_set_track_default(node->tracks + i, defaults[i]); 2.62 + node->cur_anim[1] = -1; 2.63 + 2.64 + if(!(node->animations = dynarr_alloc(1, sizeof *node->animations))) { 2.65 + return -1; 2.66 + } 2.67 + if(anm_init_animation(node->animations) == -1) { 2.68 + dynarr_free(node->animations); 2.69 + return -1; 2.70 } 2.71 2.72 - node->cache.time = ANM_TIME_INVAL; 2.73 - node->cache.inv_time = ANM_TIME_INVAL; 2.74 + /* initialize thread-local matrix cache */ 2.75 + pthread_key_create(&node->cache_key, 0); 2.76 + pthread_mutex_init(&node->cache_list_lock, 0); 2.77 + 2.78 return 0; 2.79 } 2.80 2.81 @@ -39,7 +79,17 @@ 2.82 free(node->name); 2.83 2.84 for(i=0; i<ANM_NUM_TRACKS; i++) { 2.85 - anm_destroy_track(node->tracks + i); 2.86 + anm_destroy_animation(node->animations + i); 2.87 + } 2.88 + dynarr_free(node->animations); 2.89 + 2.90 + /* destroy thread-specific cache */ 2.91 + pthread_key_delete(node->cache_key); 2.92 + 2.93 + while(node->cache_list) { 2.94 + struct mat_cache *tmp = node->cache_list; 2.95 + node->cache_list = tmp->next; 2.96 + free(tmp); 2.97 } 2.98 } 2.99 2.100 @@ -113,26 +163,6 @@ 2.101 return node->name ? node->name : ""; 2.102 } 2.103 2.104 -void anm_set_interpolator(struct anm_node *node, enum anm_interpolator in) 2.105 -{ 2.106 - int i; 2.107 - 2.108 - for(i=0; i<ANM_NUM_TRACKS; i++) { 2.109 - anm_set_track_interpolator(node->tracks + i, in); 2.110 - } 2.111 - invalidate_cache(node); 2.112 -} 2.113 - 2.114 -void anm_set_extrapolator(struct anm_node *node, enum anm_extrapolator ex) 2.115 -{ 2.116 - int i; 2.117 - 2.118 - for(i=0; i<ANM_NUM_TRACKS; i++) { 2.119 - anm_set_track_extrapolator(node->tracks + i, ex); 2.120 - } 2.121 - invalidate_cache(node); 2.122 -} 2.123 - 2.124 void anm_link_node(struct anm_node *p, struct anm_node *c) 2.125 { 2.126 c->next = p->child; 2.127 @@ -165,40 +195,398 @@ 2.128 return -1; 2.129 } 2.130 2.131 +void anm_set_pivot(struct anm_node *node, vec3_t piv) 2.132 +{ 2.133 + node->pivot = piv; 2.134 +} 2.135 + 2.136 +vec3_t anm_get_pivot(struct anm_node *node) 2.137 +{ 2.138 + return node->pivot; 2.139 +} 2.140 + 2.141 + 2.142 +/* animation management */ 2.143 + 2.144 +int anm_use_node_animation(struct anm_node *node, int aidx) 2.145 +{ 2.146 + if(aidx == node->cur_anim[0] && node->cur_anim[1] == -1) { 2.147 + return 0; /* no change, no invalidation */ 2.148 + } 2.149 + 2.150 + if(aidx < 0 || aidx >= anm_get_animation_count(node)) { 2.151 + return -1; 2.152 + } 2.153 + 2.154 + node->cur_anim[0] = aidx; 2.155 + node->cur_anim[1] = -1; 2.156 + node->cur_mix = 0; 2.157 + node->blend_dur = -1; 2.158 + 2.159 + invalidate_cache(node); 2.160 + return 0; 2.161 +} 2.162 + 2.163 +int anm_use_node_animations(struct anm_node *node, int aidx, int bidx, float t) 2.164 +{ 2.165 + int num_anim; 2.166 + 2.167 + if(node->cur_anim[0] == aidx && node->cur_anim[1] == bidx && 2.168 + fabs(t - node->cur_mix) < 1e-6) { 2.169 + return 0; /* no change, no invalidation */ 2.170 + } 2.171 + 2.172 + num_anim = anm_get_animation_count(node); 2.173 + if(aidx < 0 || aidx >= num_anim) { 2.174 + return anm_use_animation(node, bidx); 2.175 + } 2.176 + if(bidx < 0 || bidx >= num_anim) { 2.177 + return anm_use_animation(node, aidx); 2.178 + } 2.179 + node->cur_anim[0] = aidx; 2.180 + node->cur_anim[1] = bidx; 2.181 + node->cur_mix = t; 2.182 + 2.183 + invalidate_cache(node); 2.184 + return 0; 2.185 +} 2.186 + 2.187 +int anm_use_animation(struct anm_node *node, int aidx) 2.188 +{ 2.189 + struct anm_node *child; 2.190 + 2.191 + if(anm_use_node_animation(node, aidx) == -1) { 2.192 + return -1; 2.193 + } 2.194 + 2.195 + child = node->child; 2.196 + while(child) { 2.197 + if(anm_use_animation(child, aidx) == -1) { 2.198 + return -1; 2.199 + } 2.200 + child = child->next; 2.201 + } 2.202 + return 0; 2.203 +} 2.204 + 2.205 +int anm_use_animations(struct anm_node *node, int aidx, int bidx, float t) 2.206 +{ 2.207 + struct anm_node *child; 2.208 + 2.209 + if(anm_use_node_animations(node, aidx, bidx, t) == -1) { 2.210 + return -1; 2.211 + } 2.212 + 2.213 + child = node->child; 2.214 + while(child) { 2.215 + if(anm_use_animations(child, aidx, bidx, t) == -1) { 2.216 + return -1; 2.217 + } 2.218 + child = child->next; 2.219 + } 2.220 + return 0; 2.221 + 2.222 +} 2.223 + 2.224 +void anm_set_node_animation_offset(struct anm_node *node, anm_time_t offs, int which) 2.225 +{ 2.226 + if(which < 0 || which >= 2) { 2.227 + return; 2.228 + } 2.229 + node->cur_anim_offset[which] = offs; 2.230 +} 2.231 + 2.232 +anm_time_t anm_get_animation_offset(const struct anm_node *node, int which) 2.233 +{ 2.234 + if(which < 0 || which >= 2) { 2.235 + return 0; 2.236 + } 2.237 + return node->cur_anim_offset[which]; 2.238 +} 2.239 + 2.240 +void anm_set_animation_offset(struct anm_node *node, anm_time_t offs, int which) 2.241 +{ 2.242 + struct anm_node *c = node->child; 2.243 + while(c) { 2.244 + anm_set_animation_offset(c, offs, which); 2.245 + c = c->next; 2.246 + } 2.247 + 2.248 + anm_set_node_animation_offset(node, offs, which); 2.249 +} 2.250 + 2.251 +int anm_get_active_animation_index(const struct anm_node *node, int which) 2.252 +{ 2.253 + if(which < 0 || which >= 2) return -1; 2.254 + return node->cur_anim[which]; 2.255 +} 2.256 + 2.257 +struct anm_animation *anm_get_active_animation(const struct anm_node *node, int which) 2.258 +{ 2.259 + int idx = anm_get_active_animation_index(node, which); 2.260 + if(idx < 0 || idx >= anm_get_animation_count(node)) { 2.261 + return 0; 2.262 + } 2.263 + return node->animations + idx; 2.264 +} 2.265 + 2.266 +float anm_get_active_animation_mix(const struct anm_node *node) 2.267 +{ 2.268 + return node->cur_mix; 2.269 +} 2.270 + 2.271 +int anm_get_animation_count(const struct anm_node *node) 2.272 +{ 2.273 + return dynarr_size(node->animations); 2.274 +} 2.275 + 2.276 +int anm_add_node_animation(struct anm_node *node) 2.277 +{ 2.278 + struct anm_animation newanim; 2.279 + anm_init_animation(&newanim); 2.280 + 2.281 + node->animations = dynarr_push(node->animations, &newanim); 2.282 + return 0; 2.283 +} 2.284 + 2.285 +int anm_remove_node_animation(struct anm_node *node, int idx) 2.286 +{ 2.287 + fprintf(stderr, "anm_remove_animation: unimplemented!"); 2.288 + abort(); 2.289 + return 0; 2.290 +} 2.291 + 2.292 +int anm_add_animation(struct anm_node *node) 2.293 +{ 2.294 + struct anm_node *child; 2.295 + 2.296 + if(anm_add_node_animation(node) == -1) { 2.297 + return -1; 2.298 + } 2.299 + 2.300 + child = node->child; 2.301 + while(child) { 2.302 + if(anm_add_animation(child)) { 2.303 + return -1; 2.304 + } 2.305 + child = child->next; 2.306 + } 2.307 + return 0; 2.308 +} 2.309 + 2.310 +int anm_remove_animation(struct anm_node *node, int idx) 2.311 +{ 2.312 + struct anm_node *child; 2.313 + 2.314 + if(anm_remove_node_animation(node, idx) == -1) { 2.315 + return -1; 2.316 + } 2.317 + 2.318 + child = node->child; 2.319 + while(child) { 2.320 + if(anm_remove_animation(child, idx) == -1) { 2.321 + return -1; 2.322 + } 2.323 + child = child->next; 2.324 + } 2.325 + return 0; 2.326 +} 2.327 + 2.328 +struct anm_animation *anm_get_animation(struct anm_node *node, int idx) 2.329 +{ 2.330 + if(idx < 0 || idx > anm_get_animation_count(node)) { 2.331 + return 0; 2.332 + } 2.333 + return node->animations + idx; 2.334 +} 2.335 + 2.336 +struct anm_animation *anm_get_animation_by_name(struct anm_node *node, const char *name) 2.337 +{ 2.338 + return anm_get_animation(node, anm_find_animation(node, name)); 2.339 +} 2.340 + 2.341 +int anm_find_animation(struct anm_node *node, const char *name) 2.342 +{ 2.343 + int i, count = anm_get_animation_count(node); 2.344 + for(i=0; i<count; i++) { 2.345 + if(strcmp(node->animations[i].name, name) == 0) { 2.346 + return i; 2.347 + } 2.348 + } 2.349 + return -1; 2.350 +} 2.351 + 2.352 +/* all the rest act on the current animation(s) */ 2.353 + 2.354 +void anm_set_interpolator(struct anm_node *node, enum anm_interpolator in) 2.355 +{ 2.356 + int i; 2.357 + struct anm_animation *anim = anm_get_active_animation(node, 0); 2.358 + if(!anim) return; 2.359 + 2.360 + for(i=0; i<ANM_NUM_TRACKS; i++) { 2.361 + anm_set_track_interpolator(anim->tracks + i, in); 2.362 + } 2.363 + invalidate_cache(node); 2.364 +} 2.365 + 2.366 +void anm_set_extrapolator(struct anm_node *node, enum anm_extrapolator ex) 2.367 +{ 2.368 + int i; 2.369 + struct anm_animation *anim = anm_get_active_animation(node, 0); 2.370 + if(!anim) return; 2.371 + 2.372 + for(i=0; i<ANM_NUM_TRACKS; i++) { 2.373 + anm_set_track_extrapolator(anim->tracks + i, ex); 2.374 + } 2.375 + invalidate_cache(node); 2.376 +} 2.377 + 2.378 +void anm_set_node_active_animation_name(struct anm_node *node, const char *name) 2.379 +{ 2.380 + struct anm_animation *anim = anm_get_active_animation(node, 0); 2.381 + if(!anim) return; 2.382 + 2.383 + anm_set_animation_name(anim, name); 2.384 +} 2.385 + 2.386 +void anm_set_active_animation_name(struct anm_node *node, const char *name) 2.387 +{ 2.388 + struct anm_node *child; 2.389 + 2.390 + anm_set_node_active_animation_name(node, name); 2.391 + 2.392 + child = node->child; 2.393 + while(child) { 2.394 + anm_set_active_animation_name(child, name); 2.395 + child = child->next; 2.396 + } 2.397 +} 2.398 + 2.399 +const char *anm_get_active_animation_name(struct anm_node *node) 2.400 +{ 2.401 + struct anm_animation *anim = anm_get_active_animation(node, 0); 2.402 + if(anim) { 2.403 + return anim->name; 2.404 + } 2.405 + return 0; 2.406 +} 2.407 + 2.408 +/* ---- high level animation blending ---- */ 2.409 +void anm_transition(struct anm_node *node, int anmidx, anm_time_t start, anm_time_t dur) 2.410 +{ 2.411 + struct anm_node *c = node->child; 2.412 + 2.413 + if(anmidx == node->cur_anim[0]) { 2.414 + return; 2.415 + } 2.416 + 2.417 + while(c) { 2.418 + anm_transition(c, anmidx, start, dur); 2.419 + c = c->next; 2.420 + } 2.421 + 2.422 + anm_node_transition(node, anmidx, start, dur); 2.423 +} 2.424 + 2.425 +void anm_node_transition(struct anm_node *node, int anmidx, anm_time_t start, anm_time_t dur) 2.426 +{ 2.427 + if(anmidx == node->cur_anim[0]) { 2.428 + return; 2.429 + } 2.430 + 2.431 + node->cur_anim[1] = anmidx; 2.432 + node->cur_anim_offset[1] = start; 2.433 + node->blend_dur = dur; 2.434 +} 2.435 + 2.436 + 2.437 +#define BLEND_START_TM node->cur_anim_offset[1] 2.438 + 2.439 +static anm_time_t animation_time(struct anm_node *node, anm_time_t tm, int which) 2.440 +{ 2.441 + float t; 2.442 + 2.443 + if(node->blend_dur >= 0) { 2.444 + /* we're in transition... */ 2.445 + t = (float)(tm - BLEND_START_TM) / (float)node->blend_dur; 2.446 + if(t < 0.0) t = 0.0; 2.447 + 2.448 + node->cur_mix = t; 2.449 + 2.450 + if(t > 1.0) { 2.451 + /* switch completely over to the target animation and stop blending */ 2.452 + anm_use_node_animation(node, node->cur_anim[1]); 2.453 + node->cur_anim_offset[0] = node->cur_anim_offset[1]; 2.454 + } 2.455 + } 2.456 + 2.457 + return tm - node->cur_anim_offset[which]; 2.458 +} 2.459 + 2.460 + 2.461 void anm_set_position(struct anm_node *node, vec3_t pos, anm_time_t tm) 2.462 { 2.463 - anm_set_value(node->tracks + ANM_TRACK_POS_X, tm, pos.x); 2.464 - anm_set_value(node->tracks + ANM_TRACK_POS_Y, tm, pos.y); 2.465 - anm_set_value(node->tracks + ANM_TRACK_POS_Z, tm, pos.z); 2.466 + struct anm_animation *anim = anm_get_active_animation(node, 0); 2.467 + if(!anim) return; 2.468 + 2.469 + anm_set_value(anim->tracks + ANM_TRACK_POS_X, tm, pos.x); 2.470 + anm_set_value(anim->tracks + ANM_TRACK_POS_Y, tm, pos.y); 2.471 + anm_set_value(anim->tracks + ANM_TRACK_POS_Z, tm, pos.z); 2.472 invalidate_cache(node); 2.473 } 2.474 2.475 + 2.476 vec3_t anm_get_node_position(struct anm_node *node, anm_time_t tm) 2.477 { 2.478 vec3_t v; 2.479 - v.x = anm_get_value(node->tracks + ANM_TRACK_POS_X, tm); 2.480 - v.y = anm_get_value(node->tracks + ANM_TRACK_POS_Y, tm); 2.481 - v.z = anm_get_value(node->tracks + ANM_TRACK_POS_Z, tm); 2.482 + anm_time_t tm0 = animation_time(node, tm, 0); 2.483 + struct anm_animation *anim0 = anm_get_active_animation(node, 0); 2.484 + struct anm_animation *anim1 = anm_get_active_animation(node, 1); 2.485 + 2.486 + if(!anim0) { 2.487 + return v3_cons(0, 0, 0); 2.488 + } 2.489 + 2.490 + v.x = anm_get_value(anim0->tracks + ANM_TRACK_POS_X, tm0); 2.491 + v.y = anm_get_value(anim0->tracks + ANM_TRACK_POS_Y, tm0); 2.492 + v.z = anm_get_value(anim0->tracks + ANM_TRACK_POS_Z, tm0); 2.493 + 2.494 + if(anim1) { 2.495 + vec3_t v1; 2.496 + anm_time_t tm1 = animation_time(node, tm, 1); 2.497 + v1.x = anm_get_value(anim1->tracks + ANM_TRACK_POS_X, tm1); 2.498 + v1.y = anm_get_value(anim1->tracks + ANM_TRACK_POS_Y, tm1); 2.499 + v1.z = anm_get_value(anim1->tracks + ANM_TRACK_POS_Z, tm1); 2.500 + 2.501 + v.x = v.x + (v1.x - v.x) * node->cur_mix; 2.502 + v.y = v.y + (v1.y - v.y) * node->cur_mix; 2.503 + v.z = v.z + (v1.z - v.z) * node->cur_mix; 2.504 + } 2.505 + 2.506 return v; 2.507 } 2.508 2.509 void anm_set_rotation(struct anm_node *node, quat_t rot, anm_time_t tm) 2.510 { 2.511 - anm_set_value(node->tracks + ANM_TRACK_ROT_X, tm, rot.x); 2.512 - anm_set_value(node->tracks + ANM_TRACK_ROT_Y, tm, rot.y); 2.513 - anm_set_value(node->tracks + ANM_TRACK_ROT_Z, tm, rot.z); 2.514 - anm_set_value(node->tracks + ANM_TRACK_ROT_W, tm, rot.w); 2.515 + struct anm_animation *anim = anm_get_active_animation(node, 0); 2.516 + if(!anim) return; 2.517 + 2.518 + anm_set_value(anim->tracks + ANM_TRACK_ROT_X, tm, rot.x); 2.519 + anm_set_value(anim->tracks + ANM_TRACK_ROT_Y, tm, rot.y); 2.520 + anm_set_value(anim->tracks + ANM_TRACK_ROT_Z, tm, rot.z); 2.521 + anm_set_value(anim->tracks + ANM_TRACK_ROT_W, tm, rot.w); 2.522 invalidate_cache(node); 2.523 } 2.524 2.525 -quat_t anm_get_node_rotation(struct anm_node *node, anm_time_t tm) 2.526 +static quat_t get_node_rotation(struct anm_node *node, anm_time_t tm, struct anm_animation *anim) 2.527 { 2.528 #ifndef ROT_USE_SLERP 2.529 quat_t q; 2.530 - q.x = anm_get_value(node->tracks + ANM_TRACK_ROT_X, tm); 2.531 - q.y = anm_get_value(node->tracks + ANM_TRACK_ROT_Y, tm); 2.532 - q.z = anm_get_value(node->tracks + ANM_TRACK_ROT_Z, tm); 2.533 - q.w = anm_get_value(node->tracks + ANM_TRACK_ROT_W, tm); 2.534 + q.x = anm_get_value(anim->tracks + ANM_TRACK_ROT_X, tm); 2.535 + q.y = anm_get_value(anim->tracks + ANM_TRACK_ROT_Y, tm); 2.536 + q.z = anm_get_value(anim->tracks + ANM_TRACK_ROT_Z, tm); 2.537 + q.w = anm_get_value(anim->tracks + ANM_TRACK_ROT_W, tm); 2.538 return q; 2.539 #else 2.540 int idx0, idx1, last_idx; 2.541 @@ -207,10 +595,10 @@ 2.542 struct anm_track *track_x, *track_y, *track_z, *track_w; 2.543 quat_t q, q1, q2; 2.544 2.545 - track_x = node->tracks + ANM_TRACK_ROT_X; 2.546 - track_y = node->tracks + ANM_TRACK_ROT_Y; 2.547 - track_z = node->tracks + ANM_TRACK_ROT_Z; 2.548 - track_w = node->tracks + ANM_TRACK_ROT_W; 2.549 + track_x = anim->tracks + ANM_TRACK_ROT_X; 2.550 + track_y = anim->tracks + ANM_TRACK_ROT_Y; 2.551 + track_z = anim->tracks + ANM_TRACK_ROT_Z; 2.552 + track_w = anim->tracks + ANM_TRACK_ROT_W; 2.553 2.554 if(!track_x->count) { 2.555 q.x = track_x->def_val; 2.556 @@ -267,20 +655,66 @@ 2.557 #endif 2.558 } 2.559 2.560 +quat_t anm_get_node_rotation(struct anm_node *node, anm_time_t tm) 2.561 +{ 2.562 + quat_t q; 2.563 + anm_time_t tm0 = animation_time(node, tm, 0); 2.564 + struct anm_animation *anim0 = anm_get_active_animation(node, 0); 2.565 + struct anm_animation *anim1 = anm_get_active_animation(node, 1); 2.566 + 2.567 + if(!anim0) { 2.568 + return quat_identity(); 2.569 + } 2.570 + 2.571 + q = get_node_rotation(node, tm0, anim0); 2.572 + 2.573 + if(anim1) { 2.574 + anm_time_t tm1 = animation_time(node, tm, 1); 2.575 + quat_t q1 = get_node_rotation(node, tm1, anim1); 2.576 + 2.577 + q = quat_slerp(q, q1, node->cur_mix); 2.578 + } 2.579 + return q; 2.580 +} 2.581 + 2.582 void anm_set_scaling(struct anm_node *node, vec3_t scl, anm_time_t tm) 2.583 { 2.584 - anm_set_value(node->tracks + ANM_TRACK_SCL_X, tm, scl.x); 2.585 - anm_set_value(node->tracks + ANM_TRACK_SCL_Y, tm, scl.y); 2.586 - anm_set_value(node->tracks + ANM_TRACK_SCL_Z, tm, scl.z); 2.587 + struct anm_animation *anim = anm_get_active_animation(node, 0); 2.588 + if(!anim) return; 2.589 + 2.590 + anm_set_value(anim->tracks + ANM_TRACK_SCL_X, tm, scl.x); 2.591 + anm_set_value(anim->tracks + ANM_TRACK_SCL_Y, tm, scl.y); 2.592 + anm_set_value(anim->tracks + ANM_TRACK_SCL_Z, tm, scl.z); 2.593 invalidate_cache(node); 2.594 } 2.595 2.596 vec3_t anm_get_node_scaling(struct anm_node *node, anm_time_t tm) 2.597 { 2.598 vec3_t v; 2.599 - v.x = anm_get_value(node->tracks + ANM_TRACK_SCL_X, tm); 2.600 - v.y = anm_get_value(node->tracks + ANM_TRACK_SCL_Y, tm); 2.601 - v.z = anm_get_value(node->tracks + ANM_TRACK_SCL_Z, tm); 2.602 + anm_time_t tm0 = animation_time(node, tm, 0); 2.603 + struct anm_animation *anim0 = anm_get_active_animation(node, 0); 2.604 + struct anm_animation *anim1 = anm_get_active_animation(node, 1); 2.605 + 2.606 + if(!anim0) { 2.607 + return v3_cons(1, 1, 1); 2.608 + } 2.609 + 2.610 + v.x = anm_get_value(anim0->tracks + ANM_TRACK_SCL_X, tm0); 2.611 + v.y = anm_get_value(anim0->tracks + ANM_TRACK_SCL_Y, tm0); 2.612 + v.z = anm_get_value(anim0->tracks + ANM_TRACK_SCL_Z, tm0); 2.613 + 2.614 + if(anim1) { 2.615 + vec3_t v1; 2.616 + anm_time_t tm1 = animation_time(node, tm, 1); 2.617 + v1.x = anm_get_value(anim1->tracks + ANM_TRACK_SCL_X, tm1); 2.618 + v1.y = anm_get_value(anim1->tracks + ANM_TRACK_SCL_Y, tm1); 2.619 + v1.z = anm_get_value(anim1->tracks + ANM_TRACK_SCL_Z, tm1); 2.620 + 2.621 + v.x = v.x + (v1.x - v.x) * node->cur_mix; 2.622 + v.y = v.y + (v1.y - v.y) * node->cur_mix; 2.623 + v.z = v.z + (v1.z - v.z) * node->cur_mix; 2.624 + } 2.625 + 2.626 return v; 2.627 } 2.628 2.629 @@ -324,16 +758,6 @@ 2.630 return v3_mul(s, ps); 2.631 } 2.632 2.633 -void anm_set_pivot(struct anm_node *node, vec3_t piv) 2.634 -{ 2.635 - node->pivot = piv; 2.636 -} 2.637 - 2.638 -vec3_t anm_get_pivot(struct anm_node *node) 2.639 -{ 2.640 - return node->pivot; 2.641 -} 2.642 - 2.643 void anm_get_node_matrix(struct anm_node *node, mat4_t mat, anm_time_t tm) 2.644 { 2.645 int i; 2.646 @@ -371,43 +795,102 @@ 2.647 m4_inverse(mat, tmp); 2.648 } 2.649 2.650 +void anm_eval_node(struct anm_node *node, anm_time_t tm) 2.651 +{ 2.652 + anm_get_node_matrix(node, node->matrix, tm); 2.653 +} 2.654 + 2.655 +void anm_eval(struct anm_node *node, anm_time_t tm) 2.656 +{ 2.657 + struct anm_node *c; 2.658 + 2.659 + anm_eval_node(node, tm); 2.660 + 2.661 + if(node->parent) { 2.662 + /* due to post-order traversal, the parent matrix is already evaluated */ 2.663 + m4_mult(node->matrix, node->parent->matrix, node->matrix); 2.664 + } 2.665 + 2.666 + /* recersively evaluate all children */ 2.667 + c = node->child; 2.668 + while(c) { 2.669 + anm_eval(c, tm); 2.670 + c = c->next; 2.671 + } 2.672 +} 2.673 + 2.674 void anm_get_matrix(struct anm_node *node, mat4_t mat, anm_time_t tm) 2.675 { 2.676 - if(node->cache.time != tm) { 2.677 - anm_get_node_matrix(node, node->cache.matrix, tm); 2.678 + struct mat_cache *cache = pthread_getspecific(node->cache_key); 2.679 + if(!cache) { 2.680 + cache = malloc(sizeof *cache); 2.681 + assert(cache); 2.682 + 2.683 + pthread_mutex_lock(&node->cache_list_lock); 2.684 + cache->next = node->cache_list; 2.685 + node->cache_list = cache; 2.686 + pthread_mutex_unlock(&node->cache_list_lock); 2.687 + 2.688 + cache->time = ANM_TIME_INVAL; 2.689 + cache->inv_time = ANM_TIME_INVAL; 2.690 + pthread_setspecific(node->cache_key, cache); 2.691 + } 2.692 + 2.693 + if(cache->time != tm) { 2.694 + anm_get_node_matrix(node, cache->matrix, tm); 2.695 2.696 if(node->parent) { 2.697 mat4_t parent_mat; 2.698 2.699 anm_get_matrix(node->parent, parent_mat, tm); 2.700 - m4_mult(node->cache.matrix, parent_mat, node->cache.matrix); 2.701 + m4_mult(cache->matrix, parent_mat, cache->matrix); 2.702 } 2.703 - node->cache.time = tm; 2.704 + cache->time = tm; 2.705 } 2.706 - m4_copy(mat, node->cache.matrix); 2.707 + m4_copy(mat, cache->matrix); 2.708 } 2.709 2.710 void anm_get_inv_matrix(struct anm_node *node, mat4_t mat, anm_time_t tm) 2.711 { 2.712 - if(node->cache.inv_time != tm) { 2.713 + struct mat_cache *cache = pthread_getspecific(node->cache_key); 2.714 + if(!cache) { 2.715 + cache = malloc(sizeof *cache); 2.716 + assert(cache); 2.717 + 2.718 + pthread_mutex_lock(&node->cache_list_lock); 2.719 + cache->next = node->cache_list; 2.720 + node->cache_list = cache; 2.721 + pthread_mutex_unlock(&node->cache_list_lock); 2.722 + 2.723 + cache->inv_time = ANM_TIME_INVAL; 2.724 + cache->inv_time = ANM_TIME_INVAL; 2.725 + pthread_setspecific(node->cache_key, cache); 2.726 + } 2.727 + 2.728 + if(cache->inv_time != tm) { 2.729 anm_get_matrix(node, mat, tm); 2.730 - m4_inverse(node->cache.inv_matrix, mat); 2.731 - node->cache.inv_time = tm; 2.732 + m4_inverse(cache->inv_matrix, mat); 2.733 + cache->inv_time = tm; 2.734 } 2.735 - m4_copy(mat, node->cache.inv_matrix); 2.736 + m4_copy(mat, cache->inv_matrix); 2.737 } 2.738 2.739 anm_time_t anm_get_start_time(struct anm_node *node) 2.740 { 2.741 - int i; 2.742 + int i, j; 2.743 struct anm_node *c; 2.744 anm_time_t res = LONG_MAX; 2.745 2.746 - for(i=0; i<ANM_NUM_TRACKS; i++) { 2.747 - if(node->tracks[i].count) { 2.748 - anm_time_t tm = node->tracks[i].keys[0].time; 2.749 - if(tm < res) { 2.750 - res = tm; 2.751 + for(j=0; j<2; j++) { 2.752 + struct anm_animation *anim = anm_get_active_animation(node, j); 2.753 + if(!anim) break; 2.754 + 2.755 + for(i=0; i<ANM_NUM_TRACKS; i++) { 2.756 + if(anim->tracks[i].count) { 2.757 + anm_time_t tm = anim->tracks[i].keys[0].time; 2.758 + if(tm < res) { 2.759 + res = tm; 2.760 + } 2.761 } 2.762 } 2.763 } 2.764 @@ -425,15 +908,20 @@ 2.765 2.766 anm_time_t anm_get_end_time(struct anm_node *node) 2.767 { 2.768 - int i; 2.769 + int i, j; 2.770 struct anm_node *c; 2.771 anm_time_t res = LONG_MIN; 2.772 2.773 - for(i=0; i<ANM_NUM_TRACKS; i++) { 2.774 - if(node->tracks[i].count) { 2.775 - anm_time_t tm = node->tracks[i].keys[node->tracks[i].count - 1].time; 2.776 - if(tm > res) { 2.777 - res = tm; 2.778 + for(j=0; j<2; j++) { 2.779 + struct anm_animation *anim = anm_get_active_animation(node, j); 2.780 + if(!anim) break; 2.781 + 2.782 + for(i=0; i<ANM_NUM_TRACKS; i++) { 2.783 + if(anim->tracks[i].count) { 2.784 + anm_time_t tm = anim->tracks[i].keys[anim->tracks[i].count - 1].time; 2.785 + if(tm > res) { 2.786 + res = tm; 2.787 + } 2.788 } 2.789 } 2.790 } 2.791 @@ -451,5 +939,8 @@ 2.792 2.793 static void invalidate_cache(struct anm_node *node) 2.794 { 2.795 - node->cache.time = node->cache.inv_time = ANM_TIME_INVAL; 2.796 + struct mat_cache *cache = pthread_getspecific(node->cache_key); 2.797 + if(cache) { 2.798 + cache->time = cache->inv_time = ANM_TIME_INVAL; 2.799 + } 2.800 }
3.1 --- a/libs/anim/anim.h Fri Jan 17 18:16:09 2014 +0200 3.2 +++ b/libs/anim/anim.h Fri Jan 17 18:27:47 2014 +0200 3.3 @@ -3,6 +3,8 @@ 3.4 3.5 #include "config.h" 3.6 3.7 +#include <pthread.h> 3.8 + 3.9 #include <vmath/vector.h> 3.10 #include <vmath/quat.h> 3.11 #include <vmath/matrix.h> 3.12 @@ -25,17 +27,35 @@ 3.13 ANM_NUM_TRACKS 3.14 }; 3.15 3.16 +struct anm_animation { 3.17 + char *name; 3.18 + struct anm_track tracks[ANM_NUM_TRACKS]; 3.19 +}; 3.20 + 3.21 struct anm_node { 3.22 char *name; 3.23 3.24 - struct anm_track tracks[ANM_NUM_TRACKS]; 3.25 + int cur_anim[2]; 3.26 + anm_time_t cur_anim_offset[2]; 3.27 + float cur_mix; 3.28 + 3.29 + /* high-level animation blending transition duration */ 3.30 + anm_time_t blend_dur; 3.31 + 3.32 + struct anm_animation *animations; 3.33 vec3_t pivot; 3.34 3.35 /* matrix cache */ 3.36 struct mat_cache { 3.37 mat4_t matrix, inv_matrix; 3.38 anm_time_t time, inv_time; 3.39 - } cache; 3.40 + struct mat_cache *next; 3.41 + } *cache_list; 3.42 + pthread_key_t cache_key; 3.43 + pthread_mutex_t cache_list_lock; 3.44 + 3.45 + /* matrix calculated by anm_eval functions (no locking, meant as a pre-pass) */ 3.46 + mat4_t matrix; 3.47 3.48 struct anm_node *parent; 3.49 struct anm_node *child; 3.50 @@ -46,6 +66,14 @@ 3.51 extern "C" { 3.52 #endif 3.53 3.54 +int anm_init_animation(struct anm_animation *anim); 3.55 +void anm_destroy_animation(struct anm_animation *anim); 3.56 + 3.57 +void anm_set_animation_name(struct anm_animation *anim, const char *name); 3.58 + 3.59 + 3.60 +/* ---- node/hierarchy management ---- */ 3.61 + 3.62 /* node constructor and destructor */ 3.63 int anm_init_node(struct anm_node *node); 3.64 void anm_destroy_node(struct anm_node *node); 3.65 @@ -66,13 +94,82 @@ 3.66 int anm_set_node_name(struct anm_node *node, const char *name); 3.67 const char *anm_get_node_name(struct anm_node *node); 3.68 3.69 -void anm_set_interpolator(struct anm_node *node, enum anm_interpolator in); 3.70 -void anm_set_extrapolator(struct anm_node *node, enum anm_extrapolator ex); 3.71 - 3.72 /* link and unlink nodes with parent/child relations */ 3.73 void anm_link_node(struct anm_node *parent, struct anm_node *child); 3.74 int anm_unlink_node(struct anm_node *parent, struct anm_node *child); 3.75 3.76 +void anm_set_pivot(struct anm_node *node, vec3_t pivot); 3.77 +vec3_t anm_get_pivot(struct anm_node *node); 3.78 + 3.79 +/* ---- multiple animations and animation blending ---- */ 3.80 + 3.81 +/* set active animation(s) */ 3.82 +int anm_use_node_animation(struct anm_node *node, int aidx); 3.83 +int anm_use_node_animations(struct anm_node *node, int aidx, int bidx, float t); 3.84 +/* recursive variants */ 3.85 +int anm_use_animation(struct anm_node *node, int aidx); 3.86 +int anm_use_animations(struct anm_node *node, int aidx, int bidx, float t); 3.87 + 3.88 +/* set/get current animation offset(s) */ 3.89 +void anm_set_node_animation_offset(struct anm_node *node, anm_time_t offs, int which); 3.90 +anm_time_t anm_get_animation_offset(const struct anm_node *node, int which); 3.91 +/* recursive variant */ 3.92 +void anm_set_animation_offset(struct anm_node *node, anm_time_t offs, int which); 3.93 + 3.94 +/* returns the requested current animation index, which can be 0 or 1 */ 3.95 +int anm_get_active_animation_index(const struct anm_node *node, int which); 3.96 +/* returns the requested current animation, which can be 0 or 1 */ 3.97 +struct anm_animation *anm_get_active_animation(const struct anm_node *node, int which); 3.98 +float anm_get_active_animation_mix(const struct anm_node *node); 3.99 + 3.100 +int anm_get_animation_count(const struct anm_node *node); 3.101 + 3.102 +/* add/remove an animation to the specified node */ 3.103 +int anm_add_node_animation(struct anm_node *node); 3.104 +int anm_remove_node_animation(struct anm_node *node, int idx); 3.105 + 3.106 +/* add/remove an animation to the specified node and all it's descendants */ 3.107 +int anm_add_animation(struct anm_node *node); 3.108 +int anm_remove_animation(struct anm_node *node, int idx); 3.109 + 3.110 +struct anm_animation *anm_get_animation(struct anm_node *node, int idx); 3.111 +struct anm_animation *anm_get_animation_by_name(struct anm_node *node, const char *name); 3.112 + 3.113 +int anm_find_animation(struct anm_node *node, const char *name); 3.114 + 3.115 +/* set the interpolator for the (first) currently active animation */ 3.116 +void anm_set_interpolator(struct anm_node *node, enum anm_interpolator in); 3.117 +/* set the extrapolator for the (first) currently active animation */ 3.118 +void anm_set_extrapolator(struct anm_node *node, enum anm_extrapolator ex); 3.119 + 3.120 +/* set the name of the currently active animation of this node only */ 3.121 +void anm_set_node_active_animation_name(struct anm_node *node, const char *name); 3.122 +/* recursively set the name of the currently active animation for this node 3.123 + * and all it's descendants */ 3.124 +void anm_set_active_animation_name(struct anm_node *node, const char *name); 3.125 +/* get the name of the currently active animation of this node */ 3.126 +const char *anm_get_active_animation_name(struct anm_node *node); 3.127 + 3.128 + 3.129 +/* ---- high level animation blending interface ---- */ 3.130 +/* XXX this convenience interface assumes monotonically increasing time values 3.131 + * in all subsequent calls to anm_get_* and anm_eval_* functions. 3.132 + * 3.133 + * anmidx: index of the animation to transition to 3.134 + * start: when to start the transition 3.135 + * dur: transition duration 3.136 + * 3.137 + * sets up a transition from the current animation (cur_anim[0]) to another animation. 3.138 + * at time start + dur, the transition will be completed, cur_anim[0] will be the new 3.139 + * animation and cur_anim_offset[0] will be equal to start. 3.140 + */ 3.141 +void anm_transition(struct anm_node *node, int anmidx, anm_time_t start, anm_time_t dur); 3.142 +/* non-recursive variant, acts on a single node (you probably DON'T want to use this) */ 3.143 +void anm_node_transition(struct anm_node *node, int anmidx, anm_time_t start, anm_time_t dur); 3.144 + 3.145 + 3.146 +/* ---- keyframes / PRS interpolation ---- */ 3.147 + 3.148 void anm_set_position(struct anm_node *node, vec3_t pos, anm_time_t tm); 3.149 vec3_t anm_get_node_position(struct anm_node *node, anm_time_t tm); 3.150 3.151 @@ -87,13 +184,27 @@ 3.152 quat_t anm_get_rotation(struct anm_node *node, anm_time_t tm); 3.153 vec3_t anm_get_scaling(struct anm_node *node, anm_time_t tm); 3.154 3.155 -void anm_set_pivot(struct anm_node *node, vec3_t pivot); 3.156 -vec3_t anm_get_pivot(struct anm_node *node); 3.157 +/* those return the start and end times of the whole tree */ 3.158 +anm_time_t anm_get_start_time(struct anm_node *node); 3.159 +anm_time_t anm_get_end_time(struct anm_node *node); 3.160 + 3.161 + 3.162 +/* ---- transformation matrices ---- */ 3.163 3.164 /* these calculate the matrix and inverse matrix of this node alone */ 3.165 void anm_get_node_matrix(struct anm_node *node, mat4_t mat, anm_time_t tm); 3.166 void anm_get_node_inv_matrix(struct anm_node *node, mat4_t mat, anm_time_t tm); 3.167 3.168 +/* ---- top-down matrix calculation interface ---- */ 3.169 + 3.170 +/* calculate and set the matrix of this node */ 3.171 +void anm_eval_node(struct anm_node *node, anm_time_t tm); 3.172 +/* calculate and set the matrix of this node and all its children recursively */ 3.173 +void anm_eval(struct anm_node *node, anm_time_t tm); 3.174 + 3.175 + 3.176 +/* ---- bottom-up lazy matrix calculation interface ---- */ 3.177 + 3.178 /* These calculate the matrix and inverse matrix of this node taking hierarchy 3.179 * into account. The results are cached in thread-specific storage and returned 3.180 * if there's no change in time or tracks from the last query... 3.181 @@ -101,10 +212,6 @@ 3.182 void anm_get_matrix(struct anm_node *node, mat4_t mat, anm_time_t tm); 3.183 void anm_get_inv_matrix(struct anm_node *node, mat4_t mat, anm_time_t tm); 3.184 3.185 -/* those return the start and end times of the whole tree */ 3.186 -anm_time_t anm_get_start_time(struct anm_node *node); 3.187 -anm_time_t anm_get_end_time(struct anm_node *node); 3.188 - 3.189 #ifdef __cplusplus 3.190 } 3.191 #endif
4.1 --- a/libs/anim/dynarr.h Fri Jan 17 18:16:09 2014 +0200 4.2 +++ b/libs/anim/dynarr.h Fri Jan 17 18:27:47 2014 +0200 4.3 @@ -1,8 +1,6 @@ 4.4 #ifndef DYNARR_H_ 4.5 #define DYNARR_H_ 4.6 4.7 -#include "config.h" 4.8 - 4.9 void *dynarr_alloc(int elem, int szelem); 4.10 void dynarr_free(void *da); 4.11 void *dynarr_resize(void *da, int elem);
5.1 --- a/libs/anim/track.c Fri Jan 17 18:16:09 2014 +0200 5.2 +++ b/libs/anim/track.c Fri Jan 17 18:27:47 2014 +0200 5.3 @@ -69,7 +69,7 @@ 5.4 free(track); 5.5 } 5.6 5.7 -void anm_copy_track(struct anm_track *dest, struct anm_track *src) 5.8 +void anm_copy_track(struct anm_track *dest, const struct anm_track *src) 5.9 { 5.10 free(dest->name); 5.11 if(dest->keys) {
6.1 --- a/libs/anim/track.h Fri Jan 17 18:16:09 2014 +0200 6.2 +++ b/libs/anim/track.h Fri Jan 17 18:27:47 2014 +0200 6.3 @@ -59,7 +59,7 @@ 6.4 /* copies track src to dest 6.5 * XXX: dest must have been initialized first 6.6 */ 6.7 -void anm_copy_track(struct anm_track *dest, struct anm_track *src); 6.8 +void anm_copy_track(struct anm_track *dest, const struct anm_track *src); 6.9 6.10 int anm_set_track_name(struct anm_track *track, const char *name); 6.11 const char *anm_get_track_name(struct anm_track *track);
7.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 7.2 +++ b/src/goat3d_read.cc Fri Jan 17 18:27:47 2014 +0200 7.3 @@ -0,0 +1,13 @@ 7.4 +#include "goat3d_impl.h" 7.5 + 7.6 +using namespace g3dimpl; 7.7 + 7.8 +bool Scene::load(goat3d_io *io) 7.9 +{ 7.10 + return false; 7.11 +} 7.12 + 7.13 +bool Scene::load_anim(goat3d_io *io) 7.14 +{ 7.15 + return false; 7.16 +}
8.1 --- a/src/goat3d_readxml.cc Fri Jan 17 18:16:09 2014 +0200 8.2 +++ b/src/goat3d_readxml.cc Fri Jan 17 18:27:47 2014 +0200 8.3 @@ -110,6 +110,12 @@ 8.4 return true; 8.5 } 8.6 8.7 +bool Scene::load_anim_xml(goat3d_io *io) 8.8 +{ 8.9 + return false; 8.10 +} 8.11 + 8.12 + 8.13 static Material *read_material(Scene *scn, XMLElement *xml_mtl) 8.14 { 8.15 Material *mtl = new Material;
9.1 --- a/src/goat3d_write.cc Fri Jan 17 18:16:09 2014 +0200 9.2 +++ b/src/goat3d_write.cc Fri Jan 17 18:27:47 2014 +0200 9.3 @@ -225,3 +225,8 @@ 9.4 return size; 9.5 } 9.6 #endif 9.7 + 9.8 +bool Scene::save_anim(const XFormNode *node, goat3d_io *io) const 9.9 +{ 9.10 + return false; 9.11 +}
10.1 --- a/src/scene.cc Fri Jan 17 18:16:09 2014 +0200 10.2 +++ b/src/scene.cc Fri Jan 17 18:27:47 2014 +0200 10.3 @@ -204,12 +204,7 @@ 10.4 return (int)nodes.size(); 10.5 } 10.6 10.7 - 10.8 -bool Scene::load(goat3d_io *io) 10.9 -{ 10.10 - return false; 10.11 -} 10.12 - 10.13 +// Scene::load is defined in goat3d_read.cc 10.14 // Scene::loadxml is defined in goat3d_readxml.cc 10.15 // Scene::save is defined in goat3d_write.cc 10.16 // Scene::savexml is defined in goat3d_writexml.cc