gc: add toy tri-color GC implementation
As a minimum, we would like to maintain generic garbage collection of containerd resources. This includes images, bundles, containers and arbitrary content. The included implementation is just a textbook toy that we can use to inform the requirements of gc. It implements a simple, stack-based tricolor algorithm. Nothing special. Mostly, this is to ensure we think about garbage collection from the start, rather than as an afterthought or follow on feature. Signed-off-by: Stephen J Day <stephen.day@docker.com>
This commit is contained in:
		
							parent
							
								
									df9c09303c
								
							
						
					
					
						commit
						b542cc19ad
					
				
					 3 changed files with 153 additions and 0 deletions
				
			
		
							
								
								
									
										69
									
								
								gc/2
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								gc/2
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,69 @@ | |||
| package gc | ||||
| 
 | ||||
| type color uint8 | ||||
| 
 | ||||
| const ( | ||||
| 	white = iota | ||||
| 	gray | ||||
| 	black | ||||
| ) | ||||
| 
 | ||||
| func tricolor(roots []string, all []string, refs map[string][]string) []string { | ||||
| 	// all objects white -> default value of color | ||||
| 	state := make(map[string]color, len(all)) | ||||
| 	var ( | ||||
| 		grays     []string | ||||
| 		reachable []string | ||||
| 	) | ||||
| 
 | ||||
| 	grays = append(grays, roots...) | ||||
| 	// root-set objects gray. | ||||
| 	for _, ro := range roots { | ||||
| 		state[ro] = gray | ||||
| 	} | ||||
| 
 | ||||
| 	// (Repeat this as long as there are gray coloured objects) Pick a gray | ||||
| 	// object. Colour all objects referenced to from that gray object gray too. | ||||
| 	// (Except those who are black). And colour itself black. | ||||
| 
 | ||||
| 	// Pick any gray object | ||||
| 	for id := findcolor(state, gray); id != ""; id = findcolor(state, gray) { | ||||
| 		// mark all the referenced objects as gray | ||||
| 		for _, target := range refs[id] { | ||||
| 			if state[target] == white { | ||||
| 				state[target] = gray | ||||
| 			} | ||||
| 		} | ||||
| 		state[id] = black | ||||
| 	} | ||||
| 
 | ||||
| 	// All black objects are now reachable, and all white objects are | ||||
| 	// unreachable. Free those that are white! | ||||
| 	var whites []string | ||||
| 	for _, obj := range all { | ||||
| 		if state[obj] == white { | ||||
| 			whites = append(whites, obj) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return whites | ||||
| } | ||||
| 
 | ||||
| func findcolor(cs map[string]color, q color) string { | ||||
| 	// TODO(stevvooe): Super-inefficient! | ||||
| 	for id, c := range cs { | ||||
| 		if c == q { | ||||
| 			return id | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return "" | ||||
| } | ||||
| 
 | ||||
| // type colorset struct { | ||||
| // 	state map[string]color | ||||
| // } | ||||
| 
 | ||||
| // func (cs *colorset) mark(id string, c color) { | ||||
| // 	cs.state[id] = c | ||||
| // } | ||||
							
								
								
									
										54
									
								
								gc/gc.go
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								gc/gc.go
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,54 @@ | |||
| // Package gc experiments with providing central gc tooling to ensure | ||||
| // deterministic resource removaol within containerd. | ||||
| // | ||||
| // For now, we just have a single exported implementation that can be used | ||||
| // under certain use cases. | ||||
| package gc | ||||
| 
 | ||||
| // Tricolor implements basic, single-thread tri-color GC. Given the roots, the | ||||
| // complete set and a refs function, this returns the unreachable objects. | ||||
| // | ||||
| // Correct usage requires that the caller not allow the arguments to change | ||||
| // until the result is used to delete objects in the system. | ||||
| // | ||||
| // It will allocate memory proportional to the size of the reachable set. | ||||
| // | ||||
| // We can probably use this to inform a design for incremental GC by injecting | ||||
| // callbacks to the set modification algorithms. | ||||
| func Tricolor(roots []string, all []string, refs func(ref string) []string) []string { | ||||
| 	var ( | ||||
| 		grays     []string                // maintain a gray "stack" | ||||
| 		seen      = map[string]struct{}{} // or not "white", basically "seen" | ||||
| 		reachable = map[string]struct{}{} // or "block", in tri-color parlance | ||||
| 	) | ||||
| 
 | ||||
| 	grays = append(grays, roots...) | ||||
| 
 | ||||
| 	for len(grays) > 0 { | ||||
| 		// Pick any gray object | ||||
| 		id := grays[len(grays)-1] // effectively "depth first" because first element | ||||
| 		grays = grays[:len(grays)-1] | ||||
| 		seen[id] = struct{}{} // post-mark this as not-white | ||||
| 
 | ||||
| 		// mark all the referenced objects as gray | ||||
| 		for _, target := range refs(id) { | ||||
| 			if _, ok := seen[target]; !ok { | ||||
| 				grays = append(grays, target) | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		// mark as black when done | ||||
| 		reachable[id] = struct{}{} | ||||
| 	} | ||||
| 
 | ||||
| 	// All black objects are now reachable, and all white objects are | ||||
| 	// unreachable. Free those that are white! | ||||
| 	var whites []string | ||||
| 	for _, obj := range all { | ||||
| 		if _, ok := reachable[obj]; !ok { | ||||
| 			whites = append(whites, obj) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return whites | ||||
| } | ||||
							
								
								
									
										30
									
								
								gc/gc_test.go
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								gc/gc_test.go
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,30 @@ | |||
| package gc | ||||
| 
 | ||||
| import ( | ||||
| 	"reflect" | ||||
| 	"testing" | ||||
| ) | ||||
| 
 | ||||
| func TestTricolorBasic(t *testing.T) { | ||||
| 	roots := []string{"A", "C"} | ||||
| 	all := []string{"A", "B", "C", "D", "E", "F", "G"} | ||||
| 	refs := map[string][]string{ | ||||
| 		"A": {"B"}, | ||||
| 		"B": {"A"}, | ||||
| 		"C": {"D", "F", "B"}, | ||||
| 		"E": {"F", "G"}, | ||||
| 	} | ||||
| 
 | ||||
| 	unreachable := Tricolor(roots, all, lookup(refs)) | ||||
| 	expected := []string{"E", "G"} | ||||
| 
 | ||||
| 	if !reflect.DeepEqual(unreachable, expected) { | ||||
| 		t.Fatalf("incorrect unreachable set: %v != %v", unreachable, expected) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func lookup(refs map[string][]string) func(id string) []string { | ||||
| 	return func(ref string) []string { | ||||
| 		return refs[ref] | ||||
| 	} | ||||
| } | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue