Fix T82769: remove thread local data from PTCacheMem
The issue was that the same point cache was read by multiple threads at the same time (the same object was evaluated for render and for the viewport). Both threads incremented PTCacheMem->cur which lead to the crash. The fix is to remove the PTCacheMem->cur and store it on the stack instead. This way every thread has its own cur. Reviewers: brecht Differential Revision: https://developer.blender.org/D9606
This commit is contained in:
parent
e01bf7a92e
commit
dca36a8ec9
Notes:
blender-bot
2023-02-14 05:53:38 +01:00
Referenced by issue #82903, Exact Boolean with Incorrect Result - multiprecision/float round trips not exact Referenced by issue #82769, Crash rendering particle animation with particle caches [disk cache works though]
|
@ -322,9 +322,11 @@ int BKE_ptcache_data_size(int data_type);
|
|||
int BKE_ptcache_mem_index_find(struct PTCacheMem *pm, unsigned int index);
|
||||
|
||||
/* Memory cache read/write helpers. */
|
||||
void BKE_ptcache_mem_pointers_init(struct PTCacheMem *pm);
|
||||
void BKE_ptcache_mem_pointers_incr(struct PTCacheMem *pm);
|
||||
int BKE_ptcache_mem_pointers_seek(int point_index, struct PTCacheMem *pm);
|
||||
void BKE_ptcache_mem_pointers_init(struct PTCacheMem *pm, void *cur[BPHYS_TOT_DATA]);
|
||||
void BKE_ptcache_mem_pointers_incr(void *cur[BPHYS_TOT_DATA]);
|
||||
int BKE_ptcache_mem_pointers_seek(int point_index,
|
||||
struct PTCacheMem *pm,
|
||||
void *cur[BPHYS_TOT_DATA]);
|
||||
|
||||
/* Main cache reading call. */
|
||||
int BKE_ptcache_read(PTCacheID *pid, float cfra, bool no_extrapolate_old);
|
||||
|
|
|
@ -1727,27 +1727,27 @@ int BKE_ptcache_mem_index_find(PTCacheMem *pm, unsigned int index)
|
|||
return (index < pm->totpoint ? index : -1);
|
||||
}
|
||||
|
||||
void BKE_ptcache_mem_pointers_init(PTCacheMem *pm)
|
||||
void BKE_ptcache_mem_pointers_init(PTCacheMem *pm, void *cur[BPHYS_TOT_DATA])
|
||||
{
|
||||
int data_types = pm->data_types;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < BPHYS_TOT_DATA; i++) {
|
||||
pm->cur[i] = ((data_types & (1 << i)) ? pm->data[i] : NULL);
|
||||
cur[i] = ((data_types & (1 << i)) ? pm->data[i] : NULL);
|
||||
}
|
||||
}
|
||||
|
||||
void BKE_ptcache_mem_pointers_incr(PTCacheMem *pm)
|
||||
void BKE_ptcache_mem_pointers_incr(void *cur[BPHYS_TOT_DATA])
|
||||
{
|
||||
int i;
|
||||
|
||||
for (i = 0; i < BPHYS_TOT_DATA; i++) {
|
||||
if (pm->cur[i]) {
|
||||
pm->cur[i] = (char *)pm->cur[i] + ptcache_data_size[i];
|
||||
if (cur[i]) {
|
||||
cur[i] = (char *)cur[i] + ptcache_data_size[i];
|
||||
}
|
||||
}
|
||||
}
|
||||
int BKE_ptcache_mem_pointers_seek(int point_index, PTCacheMem *pm)
|
||||
int BKE_ptcache_mem_pointers_seek(int point_index, PTCacheMem *pm, void *cur[BPHYS_TOT_DATA])
|
||||
{
|
||||
int data_types = pm->data_types;
|
||||
int i, index = BKE_ptcache_mem_index_find(pm, point_index);
|
||||
|
@ -1762,7 +1762,7 @@ int BKE_ptcache_mem_pointers_seek(int point_index, PTCacheMem *pm)
|
|||
}
|
||||
|
||||
for (i = 0; i < BPHYS_TOT_DATA; i++) {
|
||||
pm->cur[i] = data_types & (1 << i) ? (char *)pm->data[i] + index * ptcache_data_size[i] : NULL;
|
||||
cur[i] = data_types & (1 << i) ? (char *)pm->data[i] + index * ptcache_data_size[i] : NULL;
|
||||
}
|
||||
|
||||
return 1;
|
||||
|
@ -1932,7 +1932,8 @@ static PTCacheMem *ptcache_disk_frame_to_mem(PTCacheID *pid, int cfra)
|
|||
}
|
||||
}
|
||||
else {
|
||||
BKE_ptcache_mem_pointers_init(pm);
|
||||
void *cur[BPHYS_TOT_DATA];
|
||||
BKE_ptcache_mem_pointers_init(pm, cur);
|
||||
ptcache_file_pointers_init(pf);
|
||||
|
||||
for (i = 0; i < pm->totpoint; i++) {
|
||||
|
@ -1940,8 +1941,8 @@ static PTCacheMem *ptcache_disk_frame_to_mem(PTCacheID *pid, int cfra)
|
|||
error = 1;
|
||||
break;
|
||||
}
|
||||
ptcache_data_copy(pf->cur, pm->cur);
|
||||
BKE_ptcache_mem_pointers_incr(pm);
|
||||
ptcache_data_copy(pf->cur, cur);
|
||||
BKE_ptcache_mem_pointers_incr(cur);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -2033,16 +2034,17 @@ static int ptcache_mem_frame_to_disk(PTCacheID *pid, PTCacheMem *pm)
|
|||
}
|
||||
}
|
||||
else {
|
||||
BKE_ptcache_mem_pointers_init(pm);
|
||||
void *cur[BPHYS_TOT_DATA];
|
||||
BKE_ptcache_mem_pointers_init(pm, cur);
|
||||
ptcache_file_pointers_init(pf);
|
||||
|
||||
for (i = 0; i < pm->totpoint; i++) {
|
||||
ptcache_data_copy(pm->cur, pf->cur);
|
||||
ptcache_data_copy(cur, pf->cur);
|
||||
if (!ptcache_file_data_write(pf)) {
|
||||
error = 1;
|
||||
break;
|
||||
}
|
||||
BKE_ptcache_mem_pointers_incr(pm);
|
||||
BKE_ptcache_mem_pointers_incr(cur);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -2160,16 +2162,17 @@ static int ptcache_read(PTCacheID *pid, int cfra)
|
|||
}
|
||||
}
|
||||
|
||||
BKE_ptcache_mem_pointers_init(pm);
|
||||
void *cur[BPHYS_TOT_DATA];
|
||||
BKE_ptcache_mem_pointers_init(pm, cur);
|
||||
|
||||
for (i = 0; i < totpoint; i++) {
|
||||
if (pm->data_types & (1 << BPHYS_DATA_INDEX)) {
|
||||
index = pm->cur[BPHYS_DATA_INDEX];
|
||||
index = cur[BPHYS_DATA_INDEX];
|
||||
}
|
||||
|
||||
pid->read_point(*index, pid->calldata, pm->cur, (float)pm->frame, NULL);
|
||||
pid->read_point(*index, pid->calldata, cur, (float)pm->frame, NULL);
|
||||
|
||||
BKE_ptcache_mem_pointers_incr(pm);
|
||||
BKE_ptcache_mem_pointers_incr(cur);
|
||||
}
|
||||
|
||||
if (pid->read_extra_data && pm->extradata.first) {
|
||||
|
@ -2216,16 +2219,16 @@ static int ptcache_interpolate(PTCacheID *pid, float cfra, int cfra1, int cfra2)
|
|||
}
|
||||
}
|
||||
|
||||
BKE_ptcache_mem_pointers_init(pm);
|
||||
void *cur[BPHYS_TOT_DATA];
|
||||
BKE_ptcache_mem_pointers_init(pm, cur);
|
||||
|
||||
for (i = 0; i < totpoint; i++) {
|
||||
if (pm->data_types & (1 << BPHYS_DATA_INDEX)) {
|
||||
index = pm->cur[BPHYS_DATA_INDEX];
|
||||
index = cur[BPHYS_DATA_INDEX];
|
||||
}
|
||||
|
||||
pid->interpolate_point(
|
||||
*index, pid->calldata, pm->cur, cfra, (float)cfra1, (float)cfra2, NULL);
|
||||
BKE_ptcache_mem_pointers_incr(pm);
|
||||
pid->interpolate_point(*index, pid->calldata, cur, cfra, (float)cfra1, (float)cfra2, NULL);
|
||||
BKE_ptcache_mem_pointers_incr(cur);
|
||||
}
|
||||
|
||||
if (pid->interpolate_extra_data && pm->extradata.first) {
|
||||
|
@ -2389,7 +2392,8 @@ static int ptcache_write(PTCacheID *pid, int cfra, int overwrite)
|
|||
pm->data_types = cfra ? pid->data_types : pid->info_types;
|
||||
|
||||
ptcache_data_alloc(pm);
|
||||
BKE_ptcache_mem_pointers_init(pm);
|
||||
void *cur[BPHYS_TOT_DATA];
|
||||
BKE_ptcache_mem_pointers_init(pm, cur);
|
||||
|
||||
if (overwrite) {
|
||||
if (cache->flag & PTCACHE_DISK_CACHE) {
|
||||
|
@ -2408,13 +2412,14 @@ static int ptcache_write(PTCacheID *pid, int cfra, int overwrite)
|
|||
|
||||
if (pid->write_point) {
|
||||
for (i = 0; i < totpoint; i++) {
|
||||
int write = pid->write_point(i, pid->calldata, pm->cur, cfra);
|
||||
int write = pid->write_point(i, pid->calldata, cur, cfra);
|
||||
if (write) {
|
||||
BKE_ptcache_mem_pointers_incr(pm);
|
||||
BKE_ptcache_mem_pointers_incr(cur);
|
||||
|
||||
void *cur2[BPHYS_TOT_DATA];
|
||||
/* newly born particles have to be copied to previous cached frame */
|
||||
if (overwrite && write == 2 && pm2 && BKE_ptcache_mem_pointers_seek(i, pm2)) {
|
||||
pid->write_point(i, pid->calldata, pm2->cur, cfra);
|
||||
if (overwrite && write == 2 && pm2 && BKE_ptcache_mem_pointers_seek(i, pm2, cur2)) {
|
||||
pid->write_point(i, pid->calldata, cur2, cfra);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -3086,8 +3091,6 @@ static PointCache *ptcache_copy(PointCache *cache, const bool copy_data)
|
|||
}
|
||||
}
|
||||
|
||||
BKE_ptcache_mem_pointers_init(pm);
|
||||
|
||||
BLI_addtail(&ncache->mem_cache, pmn);
|
||||
}
|
||||
|
||||
|
|
|
@ -5200,7 +5200,8 @@ void PE_create_particle_edit(
|
|||
|
||||
for (pm = cache->mem_cache.first; pm; pm = pm->next) {
|
||||
LOOP_POINTS {
|
||||
if (BKE_ptcache_mem_pointers_seek(p, pm) == 0) {
|
||||
void *cur[BPHYS_TOT_DATA];
|
||||
if (BKE_ptcache_mem_pointers_seek(p, pm, cur) == 0) {
|
||||
continue;
|
||||
}
|
||||
|
||||
|
@ -5212,12 +5213,12 @@ void PE_create_particle_edit(
|
|||
key = point->keys + point->totkey;
|
||||
}
|
||||
|
||||
key->co = pm->cur[BPHYS_DATA_LOCATION];
|
||||
key->vel = pm->cur[BPHYS_DATA_VELOCITY];
|
||||
key->rot = pm->cur[BPHYS_DATA_ROTATION];
|
||||
key->co = cur[BPHYS_DATA_LOCATION];
|
||||
key->vel = cur[BPHYS_DATA_VELOCITY];
|
||||
key->rot = cur[BPHYS_DATA_ROTATION];
|
||||
key->ftime = (float)pm->frame;
|
||||
key->time = &key->ftime;
|
||||
BKE_ptcache_mem_pointers_incr(pm);
|
||||
BKE_ptcache_mem_pointers_incr(cur);
|
||||
|
||||
point->totkey++;
|
||||
}
|
||||
|
|
|
@ -166,18 +166,19 @@ static void undoptcache_to_editcache(PTCacheUndo *undo, PTCacheEdit *edit)
|
|||
for (i = 0; i < BPHYS_TOT_DATA; i++) {
|
||||
pm->data[i] = MEM_dupallocN(pm->data[i]);
|
||||
}
|
||||
BKE_ptcache_mem_pointers_init(pm);
|
||||
void *cur[BPHYS_TOT_DATA];
|
||||
BKE_ptcache_mem_pointers_init(pm, cur);
|
||||
|
||||
LOOP_POINTS {
|
||||
LOOP_KEYS {
|
||||
if ((int)key->ftime == (int)pm->frame) {
|
||||
key->co = pm->cur[BPHYS_DATA_LOCATION];
|
||||
key->vel = pm->cur[BPHYS_DATA_VELOCITY];
|
||||
key->rot = pm->cur[BPHYS_DATA_ROTATION];
|
||||
key->co = cur[BPHYS_DATA_LOCATION];
|
||||
key->vel = cur[BPHYS_DATA_VELOCITY];
|
||||
key->rot = cur[BPHYS_DATA_ROTATION];
|
||||
key->time = &key->ftime;
|
||||
}
|
||||
}
|
||||
BKE_ptcache_mem_pointers_incr(pm);
|
||||
BKE_ptcache_mem_pointers_incr(cur);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -64,8 +64,6 @@ typedef struct PTCacheMem {
|
|||
|
||||
/** BPHYS_TOT_DATA. */
|
||||
void *data[8];
|
||||
/** BPHYS_TOT_DATA. */
|
||||
void *cur[8];
|
||||
|
||||
struct ListBase extradata;
|
||||
} PTCacheMem;
|
||||
|
|
Loading…
Reference in New Issue