summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJohn Glover <j@johnglover.net>2012-10-04 10:30:42 +0200
committerJohn Glover <j@johnglover.net>2012-10-04 10:30:42 +0200
commita47409a1e57e12dc296b890585aa07cddd3c245a (patch)
tree3694fe2e925ae10e9d9237172a9ab2362a345cc6
parent12fe29417d68963b1e36bc37261f7f23beb5fbea (diff)
downloadsimpl-a47409a1e57e12dc296b890585aa07cddd3c245a.tar.gz
simpl-a47409a1e57e12dc296b890585aa07cddd3c245a.tar.bz2
simpl-a47409a1e57e12dc296b890585aa07cddd3c245a.zip
[base] Bug fix: make sure that Python peak objects
are not deallocated when using Frame peaks/partials properties.
-rw-r--r--simpl/base.pyx44
1 files changed, 36 insertions, 8 deletions
diff --git a/simpl/base.pyx b/simpl/base.pyx
index 2e7ec17..ad63fdb 100644
--- a/simpl/base.pyx
+++ b/simpl/base.pyx
@@ -77,13 +77,27 @@ cdef class Frame:
def peak(self, int i):
cdef c_Peak* c_p = self.thisptr.peak(i)
- p = Peak()
- p.copy(c_p)
- return p
+ # return the same Python peak object if it exists so
+ # memory is not deallocated
+ if i < len(self._peaks) and self._peaks[i] and \
+ self._peaks[i].amplitude == c_p.amplitude and \
+ self._peaks[i].frequency == c_p.frequency and \
+ self._peaks[i].phase == c_p.phase and \
+ self._peaks[i].bandwidth == c_p.bandwidth:
+ return self._peaks[i]
+ # if not, make a new Python peak and copy the values
+ else:
+ p = Peak()
+ p.copy(c_p)
+ return p
property peaks:
def __get__(self):
- self._peaks = [self.peak(i) for i in range(self.thisptr.num_peaks())]
+ # assign updated peak list to new local variable first so that
+ # a reference to the old peak peaks is kept while updating, and
+ # memory is not deallocated
+ new_peaks = [self.peak(i) for i in range(self.thisptr.num_peaks())]
+ self._peaks = new_peaks
return self._peaks
def __set__(self, peaks):
self.thisptr.clear_peaks()
@@ -111,15 +125,29 @@ cdef class Frame:
cdef c_Peak* c_p
if not p:
c_p = self.thisptr.partial(i)
- peak = Peak()
- peak.copy(c_p)
- return peak
+ # return the same Python peak object if it exists so
+ # memory is not deallocated
+ if i < len(self._partials) and self._partials[i] and \
+ self._partials[i].amplitude == c_p.amplitude and \
+ self._partials[i].frequency == c_p.frequency and \
+ self._partials[i].phase == c_p.phase and \
+ self._partials[i].bandwidth == c_p.bandwidth:
+ return self._partials[i]
+ # if not, make a new Python peak and copy the values
+ else:
+ peak = Peak()
+ peak.copy(c_p)
+ return peak
else:
self.thisptr.partial(i, p.thisptr)
property partials:
def __get__(self):
- self._partials = [self.partial(i) for i in range(self.thisptr.num_partials())]
+ # assign updated partial list to new local variable first so that
+ # a reference to the old partial peaks is kept while updating, and
+ # memory is not deallocated
+ new_partials = [self.partial(i) for i in range(self.thisptr.num_partials())]
+ self._partials = new_partials
return self._partials
def __set__(self, peaks):
self.thisptr.clear_partials()