forked from toolshed/abra
		
	
		
			
				
	
	
		
			127 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			127 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2024 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.
 | |
| 
 | |
| //go:build race
 | |
| 
 | |
| package impl
 | |
| 
 | |
| // When running under race detector, we add a presence map of bytes, that we can access
 | |
| // in the hook functions so that we trigger the race detection whenever we have concurrent
 | |
| // Read-Writes or Write-Writes. The race detector does not otherwise detect invalid concurrent
 | |
| // access to lazy fields as all updates of bitmaps and pointers are done using atomic operations.
 | |
| type RaceDetectHookData struct {
 | |
| 	shadowPresence *[]byte
 | |
| }
 | |
| 
 | |
| // Hooks for presence bitmap operations that allocate, read and write the shadowPresence
 | |
| // using non-atomic operations.
 | |
| func (data *RaceDetectHookData) raceDetectHookAlloc(size presenceSize) {
 | |
| 	sp := make([]byte, size)
 | |
| 	atomicStoreShadowPresence(&data.shadowPresence, &sp)
 | |
| }
 | |
| 
 | |
| func (p presence) raceDetectHookPresent(num uint32) {
 | |
| 	data := p.toRaceDetectData()
 | |
| 	if data == nil {
 | |
| 		return
 | |
| 	}
 | |
| 	sp := atomicLoadShadowPresence(&data.shadowPresence)
 | |
| 	if sp != nil {
 | |
| 		_ = (*sp)[num]
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (p presence) raceDetectHookSetPresent(num uint32, size presenceSize) {
 | |
| 	data := p.toRaceDetectData()
 | |
| 	if data == nil {
 | |
| 		return
 | |
| 	}
 | |
| 	sp := atomicLoadShadowPresence(&data.shadowPresence)
 | |
| 	if sp == nil {
 | |
| 		data.raceDetectHookAlloc(size)
 | |
| 		sp = atomicLoadShadowPresence(&data.shadowPresence)
 | |
| 	}
 | |
| 	(*sp)[num] = 1
 | |
| }
 | |
| 
 | |
| func (p presence) raceDetectHookClearPresent(num uint32) {
 | |
| 	data := p.toRaceDetectData()
 | |
| 	if data == nil {
 | |
| 		return
 | |
| 	}
 | |
| 	sp := atomicLoadShadowPresence(&data.shadowPresence)
 | |
| 	if sp != nil {
 | |
| 		(*sp)[num] = 0
 | |
| 
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // raceDetectHookAllocAndCopy allocates a new shadowPresence slice at lazy and copies
 | |
| // shadowPresence bytes from src to lazy.
 | |
| func (p presence) raceDetectHookAllocAndCopy(q presence) {
 | |
| 	sData := q.toRaceDetectData()
 | |
| 	dData := p.toRaceDetectData()
 | |
| 	if sData == nil {
 | |
| 		return
 | |
| 	}
 | |
| 	srcSp := atomicLoadShadowPresence(&sData.shadowPresence)
 | |
| 	if srcSp == nil {
 | |
| 		atomicStoreShadowPresence(&dData.shadowPresence, nil)
 | |
| 		return
 | |
| 	}
 | |
| 	n := len(*srcSp)
 | |
| 	dSlice := make([]byte, n)
 | |
| 	atomicStoreShadowPresence(&dData.shadowPresence, &dSlice)
 | |
| 	for i := 0; i < n; i++ {
 | |
| 		dSlice[i] = (*srcSp)[i]
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // raceDetectHookPresent is called by the generated file interface
 | |
| // (*proto.internalFuncs) Present to optionally read an unprotected
 | |
| // shadow bitmap when race detection is enabled. In regular code it is
 | |
| // a noop.
 | |
| func raceDetectHookPresent(field *uint32, num uint32) {
 | |
| 	data := findPointerToRaceDetectData(field, num)
 | |
| 	if data == nil {
 | |
| 		return
 | |
| 	}
 | |
| 	sp := atomicLoadShadowPresence(&data.shadowPresence)
 | |
| 	if sp != nil {
 | |
| 		_ = (*sp)[num]
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // raceDetectHookSetPresent is called by the generated file interface
 | |
| // (*proto.internalFuncs) SetPresent to optionally write an unprotected
 | |
| // shadow bitmap when race detection is enabled. In regular code it is
 | |
| // a noop.
 | |
| func raceDetectHookSetPresent(field *uint32, num uint32, size presenceSize) {
 | |
| 	data := findPointerToRaceDetectData(field, num)
 | |
| 	if data == nil {
 | |
| 		return
 | |
| 	}
 | |
| 	sp := atomicLoadShadowPresence(&data.shadowPresence)
 | |
| 	if sp == nil {
 | |
| 		data.raceDetectHookAlloc(size)
 | |
| 		sp = atomicLoadShadowPresence(&data.shadowPresence)
 | |
| 	}
 | |
| 	(*sp)[num] = 1
 | |
| }
 | |
| 
 | |
| // raceDetectHookClearPresent is called by the generated file interface
 | |
| // (*proto.internalFuncs) ClearPresent to optionally write an unprotected
 | |
| // shadow bitmap when race detection is enabled. In regular code it is
 | |
| // a noop.
 | |
| func raceDetectHookClearPresent(field *uint32, num uint32) {
 | |
| 	data := findPointerToRaceDetectData(field, num)
 | |
| 	if data == nil {
 | |
| 		return
 | |
| 	}
 | |
| 	sp := atomicLoadShadowPresence(&data.shadowPresence)
 | |
| 	if sp != nil {
 | |
| 		(*sp)[num] = 0
 | |
| 	}
 | |
| }
 |