|  | // Copyright 2018 The Go Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style | 
|  | // license that can be found in the LICENSE file. | 
|  |  | 
|  | package ssa | 
|  |  | 
|  | import ( | 
|  | "cmd/internal/src" | 
|  | "math" | 
|  | ) | 
|  |  | 
|  | // A biasedSparseMap is a sparseMap for integers between J and K inclusive, | 
|  | // where J might be somewhat larger than zero (and K-J is probably much smaller than J). | 
|  | // (The motivating use case is the line numbers of statements for a single function.) | 
|  | // Not all features of a SparseMap are exported, and it is also easy to treat a | 
|  | // biasedSparseMap like a SparseSet. | 
|  | type biasedSparseMap struct { | 
|  | s     *sparseMap | 
|  | first int | 
|  | } | 
|  |  | 
|  | // newBiasedSparseMap returns a new biasedSparseMap for values between first and last, inclusive. | 
|  | func newBiasedSparseMap(first, last int) *biasedSparseMap { | 
|  | if first > last { | 
|  | return &biasedSparseMap{first: math.MaxInt32, s: nil} | 
|  | } | 
|  | return &biasedSparseMap{first: first, s: newSparseMap(1 + last - first)} | 
|  | } | 
|  |  | 
|  | // cap returns one more than the largest key valid for s | 
|  | func (s *biasedSparseMap) cap() int { | 
|  | if s == nil || s.s == nil { | 
|  | return 0 | 
|  | } | 
|  | return s.s.cap() + int(s.first) | 
|  | } | 
|  |  | 
|  | // size returns the number of entries stored in s | 
|  | func (s *biasedSparseMap) size() int { | 
|  | if s == nil || s.s == nil { | 
|  | return 0 | 
|  | } | 
|  | return s.s.size() | 
|  | } | 
|  |  | 
|  | // contains reports whether x is a key in s | 
|  | func (s *biasedSparseMap) contains(x uint) bool { | 
|  | if s == nil || s.s == nil { | 
|  | return false | 
|  | } | 
|  | if int(x) < s.first { | 
|  | return false | 
|  | } | 
|  | if int(x) >= s.cap() { | 
|  | return false | 
|  | } | 
|  | return s.s.contains(ID(int(x) - s.first)) | 
|  | } | 
|  |  | 
|  | // get returns the value s maps for key x, or -1 if | 
|  | // x is not mapped or is out of range for s. | 
|  | func (s *biasedSparseMap) get(x uint) int32 { | 
|  | if s == nil || s.s == nil { | 
|  | return -1 | 
|  | } | 
|  | if int(x) < s.first { | 
|  | return -1 | 
|  | } | 
|  | if int(x) >= s.cap() { | 
|  | return -1 | 
|  | } | 
|  | return s.s.get(ID(int(x) - s.first)) | 
|  | } | 
|  |  | 
|  | // getEntry returns the i'th key and value stored in s, | 
|  | // where 0 <= i < s.size() | 
|  | func (s *biasedSparseMap) getEntry(i int) (x uint, v int32) { | 
|  | e := s.s.contents()[i] | 
|  | x = uint(int(e.key) + s.first) | 
|  | v = e.val | 
|  | return | 
|  | } | 
|  |  | 
|  | // add inserts x->0 into s, provided that x is in the range of keys stored in s. | 
|  | func (s *biasedSparseMap) add(x uint) { | 
|  | if int(x) < s.first || int(x) >= s.cap() { | 
|  | return | 
|  | } | 
|  | s.s.set(ID(int(x)-s.first), 0, src.NoXPos) | 
|  | } | 
|  |  | 
|  | // add inserts x->v into s, provided that x is in the range of keys stored in s. | 
|  | func (s *biasedSparseMap) set(x uint, v int32) { | 
|  | if int(x) < s.first || int(x) >= s.cap() { | 
|  | return | 
|  | } | 
|  | s.s.set(ID(int(x)-s.first), v, src.NoXPos) | 
|  | } | 
|  |  | 
|  | // remove removes key x from s. | 
|  | func (s *biasedSparseMap) remove(x uint) { | 
|  | if int(x) < s.first || int(x) >= s.cap() { | 
|  | return | 
|  | } | 
|  | s.s.remove(ID(int(x) - s.first)) | 
|  | } | 
|  |  | 
|  | func (s *biasedSparseMap) clear() { | 
|  | if s.s != nil { | 
|  | s.s.clear() | 
|  | } | 
|  | } |