core/bloombits: use general filters instead of addresses and topics
This commit is contained in:
		
							parent
							
								
									6ff2c02991
								
							
						
					
					
						commit
						451ffdb62b
					
				| @ -24,7 +24,6 @@ import ( | |||||||
| 	"sync/atomic" | 	"sync/atomic" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
| 	"github.com/ethereum/go-ethereum/common" |  | ||||||
| 	"github.com/ethereum/go-ethereum/common/bitutil" | 	"github.com/ethereum/go-ethereum/common/bitutil" | ||||||
| 	"github.com/ethereum/go-ethereum/crypto" | 	"github.com/ethereum/go-ethereum/crypto" | ||||||
| ) | ) | ||||||
| @ -68,8 +67,7 @@ type Retrieval struct { | |||||||
| type Matcher struct { | type Matcher struct { | ||||||
| 	sectionSize uint64 // Size of the data batches to filter on
 | 	sectionSize uint64 // Size of the data batches to filter on
 | ||||||
| 
 | 
 | ||||||
| 	addresses  []bloomIndexes      // Addresses the system is filtering for
 | 	filters    [][]bloomIndexes    // Filter the system is matching for
 | ||||||
| 	topics     [][]bloomIndexes    // Topics the system is filtering for
 |  | ||||||
| 	schedulers map[uint]*scheduler // Retrieval schedulers for loading bloom bits
 | 	schedulers map[uint]*scheduler // Retrieval schedulers for loading bloom bits
 | ||||||
| 
 | 
 | ||||||
| 	retrievers chan chan uint       // Retriever processes waiting for bit allocations
 | 	retrievers chan chan uint       // Retriever processes waiting for bit allocations
 | ||||||
| @ -82,7 +80,8 @@ type Matcher struct { | |||||||
| 
 | 
 | ||||||
| // NewMatcher creates a new pipeline for retrieving bloom bit streams and doing
 | // NewMatcher creates a new pipeline for retrieving bloom bit streams and doing
 | ||||||
| // address and topic filtering on them.
 | // address and topic filtering on them.
 | ||||||
| func NewMatcher(sectionSize uint64, addresses []common.Address, topics [][]common.Hash) *Matcher { | func NewMatcher(sectionSize uint64, filters [][][]byte) *Matcher { | ||||||
|  | 	// Create the matcher instance
 | ||||||
| 	m := &Matcher{ | 	m := &Matcher{ | ||||||
| 		sectionSize: sectionSize, | 		sectionSize: sectionSize, | ||||||
| 		schedulers:  make(map[uint]*scheduler), | 		schedulers:  make(map[uint]*scheduler), | ||||||
| @ -91,48 +90,25 @@ func NewMatcher(sectionSize uint64, addresses []common.Address, topics [][]commo | |||||||
| 		retrievals:  make(chan chan *Retrieval), | 		retrievals:  make(chan chan *Retrieval), | ||||||
| 		deliveries:  make(chan *Retrieval), | 		deliveries:  make(chan *Retrieval), | ||||||
| 	} | 	} | ||||||
| 	m.setAddresses(addresses) | 	// Calculate the bloom bit indexes for the groups we're interested in
 | ||||||
| 	m.setTopics(topics) | 	m.filters = nil | ||||||
| 	return m |  | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| // setAddresses configures the matcher to only return logs that are generated
 | 	for _, filter := range filters { | ||||||
| // from addresses that are included in the given list.
 | 		bloomBits := make([]bloomIndexes, len(filter)) | ||||||
| func (m *Matcher) setAddresses(addresses []common.Address) { | 		for i, clause := range filter { | ||||||
| 	// Calculate the bloom bit indexes for the addresses we're interested in
 | 			bloomBits[i] = calcBloomIndexes(clause) | ||||||
| 	m.addresses = make([]bloomIndexes, len(addresses)) | 		} | ||||||
| 	for i, address := range addresses { | 		m.filters = append(m.filters, bloomBits) | ||||||
| 		m.addresses[i] = calcBloomIndexes(address.Bytes()) |  | ||||||
| 	} | 	} | ||||||
| 	// For every bit, create a scheduler to load/download the bit vectors
 | 	// For every bit, create a scheduler to load/download the bit vectors
 | ||||||
| 	for _, bloomIndexList := range m.addresses { | 	for _, bloomIndexLists := range m.filters { | ||||||
| 		for _, bloomIndex := range bloomIndexList { |  | ||||||
| 			m.addScheduler(bloomIndex) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // setTopics configures the matcher to only return logs that have topics matching
 |  | ||||||
| // the given list.
 |  | ||||||
| func (m *Matcher) setTopics(topicsList [][]common.Hash) { |  | ||||||
| 	// Calculate the bloom bit indexes for the topics we're interested in
 |  | ||||||
| 	m.topics = nil |  | ||||||
| 
 |  | ||||||
| 	for _, topics := range topicsList { |  | ||||||
| 		bloomBits := make([]bloomIndexes, len(topics)) |  | ||||||
| 		for i, topic := range topics { |  | ||||||
| 			bloomBits[i] = calcBloomIndexes(topic.Bytes()) |  | ||||||
| 		} |  | ||||||
| 		m.topics = append(m.topics, bloomBits) |  | ||||||
| 	} |  | ||||||
| 	// For every bit, create a scheduler to load/download the bit vectors
 |  | ||||||
| 	for _, bloomIndexLists := range m.topics { |  | ||||||
| 		for _, bloomIndexList := range bloomIndexLists { | 		for _, bloomIndexList := range bloomIndexLists { | ||||||
| 			for _, bloomIndex := range bloomIndexList { | 			for _, bloomIndex := range bloomIndexList { | ||||||
| 				m.addScheduler(bloomIndex) | 				m.addScheduler(bloomIndex) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 	return m | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // addScheduler adds a bit stream retrieval scheduler for the given bit index if
 | // addScheduler adds a bit stream retrieval scheduler for the given bit index if
 | ||||||
| @ -250,14 +226,10 @@ func (m *Matcher) run(begin, end uint64, buffer int, session *MatcherSession) ch | |||||||
| 		} | 		} | ||||||
| 	}() | 	}() | ||||||
| 	// Assemble the daisy-chained filtering pipeline
 | 	// Assemble the daisy-chained filtering pipeline
 | ||||||
| 	blooms := m.topics |  | ||||||
| 	if len(m.addresses) > 0 { |  | ||||||
| 		blooms = append([][]bloomIndexes{m.addresses}, blooms...) |  | ||||||
| 	} |  | ||||||
| 	next := source | 	next := source | ||||||
| 	dist := make(chan *request, buffer) | 	dist := make(chan *request, buffer) | ||||||
| 
 | 
 | ||||||
| 	for _, bloom := range blooms { | 	for _, bloom := range m.filters { | ||||||
| 		next = m.subMatch(next, dist, bloom, session) | 		next = m.subMatch(next, dist, bloom, session) | ||||||
| 	} | 	} | ||||||
| 	// Start the request distribution
 | 	// Start the request distribution
 | ||||||
|  | |||||||
| @ -94,10 +94,8 @@ func testMatcherBothModes(t *testing.T, filter [][]bloomIndexes, blocks uint64, | |||||||
| // number of requests made for cross validation between different modes.
 | // number of requests made for cross validation between different modes.
 | ||||||
| func testMatcher(t *testing.T, filter [][]bloomIndexes, blocks uint64, intermittent bool, retrievals uint32, maxReqCount int) uint32 { | func testMatcher(t *testing.T, filter [][]bloomIndexes, blocks uint64, intermittent bool, retrievals uint32, maxReqCount int) uint32 { | ||||||
| 	// Create a new matcher an simulate our explicit random bitsets
 | 	// Create a new matcher an simulate our explicit random bitsets
 | ||||||
| 	matcher := NewMatcher(testSectionSize, nil, nil) | 	matcher := NewMatcher(testSectionSize, nil) | ||||||
| 
 | 	matcher.filters = filter | ||||||
| 	matcher.addresses = filter[0] |  | ||||||
| 	matcher.topics = filter[1:] |  | ||||||
| 
 | 
 | ||||||
| 	for _, rule := range filter { | 	for _, rule := range filter { | ||||||
| 		for _, topic := range rule { | 		for _, topic := range rule { | ||||||
|  | |||||||
| @ -60,6 +60,23 @@ type Filter struct { | |||||||
| // New creates a new filter which uses a bloom filter on blocks to figure out whether
 | // New creates a new filter which uses a bloom filter on blocks to figure out whether
 | ||||||
| // a particular block is interesting or not.
 | // a particular block is interesting or not.
 | ||||||
| func New(backend Backend, begin, end int64, addresses []common.Address, topics [][]common.Hash) *Filter { | func New(backend Backend, begin, end int64, addresses []common.Address, topics [][]common.Hash) *Filter { | ||||||
|  | 	// Flatten the address and topic filter clauses into a single filter system
 | ||||||
|  | 	var filters [][][]byte | ||||||
|  | 	if len(addresses) > 0 { | ||||||
|  | 		filter := make([][]byte, len(addresses)) | ||||||
|  | 		for i, address := range addresses { | ||||||
|  | 			filter[i] = address.Bytes() | ||||||
|  | 		} | ||||||
|  | 		filters = append(filters, filter) | ||||||
|  | 	} | ||||||
|  | 	for _, topicList := range topics { | ||||||
|  | 		filter := make([][]byte, len(topicList)) | ||||||
|  | 		for i, topic := range topicList { | ||||||
|  | 			filter[i] = topic.Bytes() | ||||||
|  | 		} | ||||||
|  | 		filters = append(filters, filter) | ||||||
|  | 	} | ||||||
|  | 	// Assemble and return the filter
 | ||||||
| 	size, _ := backend.BloomStatus() | 	size, _ := backend.BloomStatus() | ||||||
| 
 | 
 | ||||||
| 	return &Filter{ | 	return &Filter{ | ||||||
| @ -69,7 +86,7 @@ func New(backend Backend, begin, end int64, addresses []common.Address, topics [ | |||||||
| 		addresses: addresses, | 		addresses: addresses, | ||||||
| 		topics:    topics, | 		topics:    topics, | ||||||
| 		db:        backend.ChainDb(), | 		db:        backend.ChainDb(), | ||||||
| 		matcher:   bloombits.NewMatcher(size, addresses, topics), | 		matcher:   bloombits.NewMatcher(size, filters), | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user