make the write lock scope limited within a function

Also avoid removing the writing set if there was an error while writing.
This commit is contained in:
vyzo 2021-08-10 10:10:12 +03:00
parent a9403b42f0
commit 79f348a01c

View File

@ -25,6 +25,7 @@ type BadgerMarkSet struct {
cond sync.Cond cond sync.Cond
pend map[string]struct{} pend map[string]struct{}
writing map[int]map[string]struct{} writing map[int]map[string]struct{}
writers int
seqno int seqno int
version int version int
@ -81,13 +82,19 @@ func (e *BadgerMarkSetEnv) Close() error {
func (s *BadgerMarkSet) Mark(c cid.Cid) error { func (s *BadgerMarkSet) Mark(c cid.Cid) error {
s.mx.Lock() s.mx.Lock()
if s.pend == nil { if s.pend == nil {
s.mx.Unlock() s.mx.Unlock()
return errMarkSetClosed return errMarkSetClosed
} }
return s.put(string(c.Hash())) write, seqno := s.put(string(c.Hash()))
s.mx.Unlock()
if write {
return s.write(seqno)
}
return nil
} }
func (s *BadgerMarkSet) Has(c cid.Cid) (bool, error) { func (s *BadgerMarkSet) Has(c cid.Cid) (bool, error) {
@ -129,7 +136,6 @@ func (s *BadgerMarkSet) Visit(c cid.Cid) (bool, error) {
s.mx.RUnlock() s.mx.RUnlock()
s.mx.Lock() s.mx.Lock()
// we have to do the check dance again // we have to do the check dance again
has, err = s.tryPending(pendKey) has, err = s.tryPending(pendKey)
if has || err != nil { if has || err != nil {
@ -146,11 +152,14 @@ func (s *BadgerMarkSet) Visit(c cid.Cid) (bool, error) {
} }
} }
if err := s.put(pendKey); err != nil { write, seqno := s.put(pendKey)
return false, err s.mx.Unlock()
if write {
err = s.write(seqno)
} }
return true, nil return true, err
} }
// reader holds the (r)lock // reader holds the (r)lock
@ -190,28 +199,43 @@ func (s *BadgerMarkSet) tryDB(key []byte) (has bool, err error) {
} }
} }
// writer holds the exclusive lock; put releases it // writer holds the exclusive lock
func (s *BadgerMarkSet) put(key string) error { func (s *BadgerMarkSet) put(key string) (write bool, seqno int) {
s.pend[key] = struct{}{} s.pend[key] = struct{}{}
if len(s.pend) < badgerMarkSetBatchSize { if len(s.pend) < badgerMarkSetBatchSize {
s.mx.Unlock() return false, 0
return nil
} }
pend := s.pend seqno = s.seqno
seqno := s.seqno
s.seqno++ s.seqno++
s.writing[seqno] = pend s.writing[seqno] = s.pend
s.pend = make(map[string]struct{}) s.pend = make(map[string]struct{})
return true, seqno
}
func (s *BadgerMarkSet) write(seqno int) (err error) {
s.mx.Lock()
if s.pend == nil {
s.mx.Unlock()
return errMarkSetClosed
}
pend := s.writing[seqno]
s.writers++
s.mx.Unlock() s.mx.Unlock()
defer func() { defer func() {
s.mx.Lock() s.mx.Lock()
defer s.mx.Unlock() defer s.mx.Unlock()
s.version++ if err == nil {
delete(s.writing, seqno) delete(s.writing, seqno)
if len(s.writing) == 0 { s.version++
}
s.writers--
if s.writers == 0 {
s.cond.Broadcast() s.cond.Broadcast()
} }
}() }()
@ -222,12 +246,12 @@ func (s *BadgerMarkSet) put(key string) error {
defer batch.Cancel() defer batch.Cancel()
for k := range pend { for k := range pend {
if err := batch.Set([]byte(k), empty); err != nil { if err = batch.Set([]byte(k), empty); err != nil {
return xerrors.Errorf("error setting batch: %w", err) return xerrors.Errorf("error setting batch: %w", err)
} }
} }
err := batch.Flush() err = batch.Flush()
if err != nil { if err != nil {
return xerrors.Errorf("error flushing batch to badger markset: %w", err) return xerrors.Errorf("error flushing batch to badger markset: %w", err)
} }
@ -243,7 +267,7 @@ func (s *BadgerMarkSet) Close() error {
return nil return nil
} }
for len(s.writing) > 0 { for s.writers > 0 {
s.cond.Wait() s.cond.Wait()
} }