* Fixed a bug in the garbage collector.
git-svn-id: svn://svn.zoy.org/elk/trunk@138 55e467fa-43c5-0310-a8a2-de718669efc6
This commit is contained in:
parent
335719f3a1
commit
2879388fa2
7
BUGS
7
BUGS
|
@ -36,10 +36,3 @@ Generational/Incremental Garbage Collector
|
||||||
|
|
||||||
o The percentage displayed at the end of a GC run is sometimes wrong.
|
o The percentage displayed at the end of a GC run is sometimes wrong.
|
||||||
|
|
||||||
o This simple test causes a valgrind error after the first GC run:
|
|
||||||
|
|
||||||
(define (fact x) (if (> x 1) (* x (fact (- x 1))) 1))
|
|
||||||
(fact 100)
|
|
||||||
(fact 1000)
|
|
||||||
(fact 1000)
|
|
||||||
|
|
||||||
|
|
|
@ -196,8 +196,7 @@ static void TerminateGC ();
|
||||||
|
|
||||||
#define IS_CLUSTER(a,b) (SAME_PHYSPAGE (PAGE_TO_ADDR ((a)), \
|
#define IS_CLUSTER(a,b) (SAME_PHYSPAGE (PAGE_TO_ADDR ((a)), \
|
||||||
PAGE_TO_ADDR ((b))) || \
|
PAGE_TO_ADDR ((b))) || \
|
||||||
(space[(a)&hp_per_pp_mask] == \
|
(space[a] == space[b] && \
|
||||||
space[((b)&hp_per_pp_mask)+hp_per_pp] && \
|
|
||||||
type[(a)&hp_per_pp_mask] == OBJECTPAGE && \
|
type[(a)&hp_per_pp_mask] == OBJECTPAGE && \
|
||||||
type[((b)&hp_per_pp_mask)+hp_per_pp] == OBJECTPAGE))
|
type[((b)&hp_per_pp_mask)+hp_per_pp] == OBJECTPAGE))
|
||||||
|
|
||||||
|
@ -805,54 +804,54 @@ static void AllocPage (pageno_t npg) {
|
||||||
|
|
||||||
/* now look for a cluster of npg free pages. cont_free counts the
|
/* now look for a cluster of npg free pages. cont_free counts the
|
||||||
* number of free pages found, first_freepage is the number of the
|
* number of free pages found, first_freepage is the number of the
|
||||||
* first free heap page in the cluster.
|
* first free heap page in the cluster. */
|
||||||
*/
|
|
||||||
|
|
||||||
for (p = spanning_pages, cont_free = 0; p; p--) {
|
for (p = spanning_pages, cont_free = 0; p; p--) {
|
||||||
if (space[current_freepage] < previous_space
|
|
||||||
&& !STABLE (current_freepage)) {
|
|
||||||
if (cont_free == 0) {
|
|
||||||
/* This is our first free page, first check that we have a
|
|
||||||
* continuous cluster of pages (we'll check later that they
|
|
||||||
* are free). Otherwise, go to the next free page */
|
|
||||||
if (current_freepage+npg-1 <= lastpage
|
|
||||||
&& IS_CLUSTER (current_freepage, current_freepage+npg-1))
|
|
||||||
first_freepage = current_freepage;
|
|
||||||
else {
|
|
||||||
current_freepage = next (current_freepage -
|
|
||||||
current_freepage % hp_per_pp +
|
|
||||||
hp_per_pp-1);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
cont_free++;
|
/* If we have more space than before, or if the current page is
|
||||||
|
* stable, start again with the next page. */
|
||||||
if (cont_free == npg) {
|
if (space[current_freepage] >= previous_space
|
||||||
space[first_freepage] = current_space;
|
|| STABLE (current_freepage)) {
|
||||||
type[first_freepage] = OBJECTPAGE;
|
|
||||||
for (n = 1; n < npg; n++) {
|
|
||||||
space[first_freepage+n] = current_space;
|
|
||||||
type[first_freepage+n] = CONTPAGE;
|
|
||||||
}
|
|
||||||
current_freep = PAGE_TO_OBJ (first_freepage);
|
|
||||||
current_free = npg*PAGEWORDS;
|
|
||||||
current_pages += npg;
|
|
||||||
allocated_pages += npg;
|
|
||||||
current_freepage = next (first_freepage+npg-1);
|
|
||||||
if (ProtectedInRegion (first_freepage, npg))
|
|
||||||
(void)ScanCluster (PHYSPAGE (first_freepage));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* check the next free page. If we warped, reset cont_free to 0. */
|
|
||||||
current_freepage = next (current_freepage);
|
|
||||||
if (current_freepage == firstpage) cont_free = 0;
|
|
||||||
|
|
||||||
} else {
|
|
||||||
current_freepage = next (current_freepage);
|
current_freepage = next (current_freepage);
|
||||||
cont_free = 0;
|
cont_free = 0;
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (cont_free == 0) {
|
||||||
|
/* This is our first free page, first check that we have a
|
||||||
|
* continuous cluster of pages (we'll check later that they
|
||||||
|
* are free). Otherwise, go to the next free page. */
|
||||||
|
if ((current_freepage+npg-1) > lastpage
|
||||||
|
|| !IS_CLUSTER (current_freepage, current_freepage+npg-1)) {
|
||||||
|
current_freepage = next ((current_freepage&hp_per_pp_mask)
|
||||||
|
+hp_per_pp-1);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
first_freepage = current_freepage;
|
||||||
|
}
|
||||||
|
|
||||||
|
cont_free++;
|
||||||
|
|
||||||
|
if (cont_free == npg) {
|
||||||
|
space[first_freepage] = current_space;
|
||||||
|
type[first_freepage] = OBJECTPAGE;
|
||||||
|
for (n = 1; n < npg; n++) {
|
||||||
|
space[first_freepage+n] = current_space;
|
||||||
|
type[first_freepage+n] = CONTPAGE;
|
||||||
|
}
|
||||||
|
current_freep = PAGE_TO_OBJ (first_freepage);
|
||||||
|
current_free = npg*PAGEWORDS;
|
||||||
|
current_pages += npg;
|
||||||
|
allocated_pages += npg;
|
||||||
|
current_freepage = next (first_freepage+npg-1);
|
||||||
|
if (ProtectedInRegion (first_freepage, npg))
|
||||||
|
(void)ScanCluster (PHYSPAGE (first_freepage));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* check the next free page. If we warped, reset cont_free to 0. */
|
||||||
|
current_freepage = next (current_freepage);
|
||||||
|
if (current_freepage == firstpage) cont_free = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* no space available, try to expand heap */
|
/* no space available, try to expand heap */
|
||||||
|
|
Loading…
Reference in New Issue