mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-11 02:18:29 -04:00
HAL_ChibiOS: prevent sharing the same DMA stream on TIMn_UP channels
this avoids an issue with DShot where we run the same DMA stream on two timers at once
This commit is contained in:
parent
6cb263056d
commit
4fbdb59398
@ -11,13 +11,17 @@ dma_map = None
|
|||||||
|
|
||||||
debug = False
|
debug = False
|
||||||
|
|
||||||
def check_possibility(periph, dma_stream, curr_dict, dma_map, check_list, cannot_use_stream):
|
def check_possibility(periph, dma_stream, curr_dict, dma_map, check_list, cannot_use_stream, forbidden_map):
|
||||||
global ignore_list
|
global ignore_list
|
||||||
if debug:
|
if debug:
|
||||||
print('............ Checking ', periph, dma_stream, 'without', cannot_use_stream)
|
print('............ Checking ', periph, dma_stream, 'without', cannot_use_stream)
|
||||||
for other_periph in curr_dict:
|
for other_periph in curr_dict:
|
||||||
if other_periph != periph:
|
if other_periph != periph:
|
||||||
if curr_dict[other_periph] == dma_stream:
|
if curr_dict[other_periph] == dma_stream:
|
||||||
|
if other_periph in forbidden_map[periph]:
|
||||||
|
if debug:
|
||||||
|
print('.................... Forbidden', periph, other_periph)
|
||||||
|
return False
|
||||||
if debug:
|
if debug:
|
||||||
print('.................... Collision', other_periph, dma_stream)
|
print('.................... Collision', other_periph, dma_stream)
|
||||||
ignore_list.append(periph)
|
ignore_list.append(periph)
|
||||||
@ -37,7 +41,8 @@ def check_possibility(periph, dma_stream, curr_dict, dma_map, check_list, cannot
|
|||||||
|
|
||||||
if stream != curr_dict[other_periph] and \
|
if stream != curr_dict[other_periph] and \
|
||||||
stream not in cannot_use_stream and \
|
stream not in cannot_use_stream and \
|
||||||
check_possibility(other_periph, stream, curr_dict, dma_map, check_list, cannot_use_stream+[(dma_stream)]):
|
check_possibility(other_periph, stream, curr_dict, dma_map, check_list,
|
||||||
|
cannot_use_stream+[(dma_stream)], forbidden_map):
|
||||||
curr_dict[other_periph] = stream
|
curr_dict[other_periph] = stream
|
||||||
if debug:
|
if debug:
|
||||||
print ('....................... Resolving', other_periph, stream)
|
print ('....................... Resolving', other_periph, stream)
|
||||||
@ -172,6 +177,7 @@ def generate_DMAMUX_map_mask(peripheral_list, channel_mask, noshare_list, dma_ex
|
|||||||
stream = ii % 8
|
stream = ii % 8
|
||||||
|
|
||||||
if (dma,stream) in dma_map[p]:
|
if (dma,stream) in dma_map[p]:
|
||||||
|
# this peripheral is already using the stream
|
||||||
continue
|
continue
|
||||||
|
|
||||||
# prevent attempts to share with other half of same peripheral
|
# prevent attempts to share with other half of same peripheral
|
||||||
@ -233,6 +239,47 @@ def generate_DMAMUX_map(peripheral_list, noshare_list, dma_exclude):
|
|||||||
print('dma_map_both: ', both)
|
print('dma_map_both: ', both)
|
||||||
return both
|
return both
|
||||||
|
|
||||||
|
def sharing_allowed(p1, p2):
|
||||||
|
'''return true if sharing is allowed between p1 and p2'''
|
||||||
|
if p1 == p2:
|
||||||
|
return True
|
||||||
|
# don't allow RX and TX of same peripheral to share
|
||||||
|
if p1.endswith('_RX') and p2.endswith('_TX') and p1[:-2] == p2[:-2]:
|
||||||
|
return False
|
||||||
|
# don't allow sharing of two TIMn_UP channels as DShot code can't cope
|
||||||
|
if p1.endswith("_UP") and p2.endswith("_UP") and p1.startswith("TIM") and p2.startswith("TIM"):
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
|
def check_sharing(shared):
|
||||||
|
'''check if DMA channel sharing is OK'''
|
||||||
|
for p in shared:
|
||||||
|
# don't share UART RX with anything
|
||||||
|
if (p.startswith("UART") or p.startswith("USART")) and p.endswith("_RX"):
|
||||||
|
print("Illegal sharing of %s" % p)
|
||||||
|
return False
|
||||||
|
|
||||||
|
# don't share ADC with anything
|
||||||
|
if p.startswith("ADC"):
|
||||||
|
print("Illegal sharing of %s" % p)
|
||||||
|
return False
|
||||||
|
|
||||||
|
for p2 in shared:
|
||||||
|
if not sharing_allowed(p, p2):
|
||||||
|
print("Illegal sharing of %s and %s" % (p, p2))
|
||||||
|
return False
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
def forbidden_list(p, peripheral_list):
|
||||||
|
'''given a peripheral, form a list of other peripherals we may not share with'''
|
||||||
|
ret = []
|
||||||
|
for p2 in peripheral_list:
|
||||||
|
if not sharing_allowed(p, p2):
|
||||||
|
ret.append(p2)
|
||||||
|
return ret
|
||||||
|
|
||||||
|
|
||||||
def write_dma_header(f, peripheral_list, mcu_type, dma_exclude=[],
|
def write_dma_header(f, peripheral_list, mcu_type, dma_exclude=[],
|
||||||
dma_priority='', dma_noshare=''):
|
dma_priority='', dma_noshare=''):
|
||||||
'''write out a DMA resolver header file'''
|
'''write out a DMA resolver header file'''
|
||||||
@ -271,6 +318,11 @@ def write_dma_header(f, peripheral_list, mcu_type, dma_exclude=[],
|
|||||||
unassigned = []
|
unassigned = []
|
||||||
curr_dict = {}
|
curr_dict = {}
|
||||||
|
|
||||||
|
# build a map from peripheral name to a list of peripherals that it cannot share with
|
||||||
|
forbidden_map = {}
|
||||||
|
for p in peripheral_list:
|
||||||
|
forbidden_map[p] = forbidden_list(p, peripheral_list)
|
||||||
|
|
||||||
for periph in peripheral_list:
|
for periph in peripheral_list:
|
||||||
if "_CH" in periph:
|
if "_CH" in periph:
|
||||||
has_bdshot = True # the list contains a CH port
|
has_bdshot = True # the list contains a CH port
|
||||||
@ -290,7 +342,7 @@ def write_dma_header(f, peripheral_list, mcu_type, dma_exclude=[],
|
|||||||
print('........Possibility for', periph, streamchan)
|
print('........Possibility for', periph, streamchan)
|
||||||
stream = (streamchan[0], streamchan[1])
|
stream = (streamchan[0], streamchan[1])
|
||||||
if check_possibility(periph, stream, curr_dict, dma_map,
|
if check_possibility(periph, stream, curr_dict, dma_map,
|
||||||
check_list, []):
|
check_list, [], forbidden_map):
|
||||||
curr_dict[periph] = stream
|
curr_dict[periph] = stream
|
||||||
if debug:
|
if debug:
|
||||||
print ('....................... Setting', periph, stream)
|
print ('....................... Setting', periph, stream)
|
||||||
@ -319,7 +371,7 @@ def write_dma_header(f, peripheral_list, mcu_type, dma_exclude=[],
|
|||||||
stream = (streamchan[0], streamchan[1])
|
stream = (streamchan[0], streamchan[1])
|
||||||
share_ok = True
|
share_ok = True
|
||||||
for periph2 in stream_assign[stream]:
|
for periph2 in stream_assign[stream]:
|
||||||
if not can_share(periph, noshare_list) or not can_share(periph2, noshare_list):
|
if not can_share(periph, noshare_list) or not can_share(periph2, noshare_list) or periph2 in forbidden_map[periph]:
|
||||||
share_ok = False
|
share_ok = False
|
||||||
if share_ok:
|
if share_ok:
|
||||||
share_possibility.append(stream)
|
share_possibility.append(stream)
|
||||||
@ -336,6 +388,12 @@ def write_dma_header(f, peripheral_list, mcu_type, dma_exclude=[],
|
|||||||
unassigned_new.remove(periph)
|
unassigned_new.remove(periph)
|
||||||
unassigned = unassigned_new
|
unassigned = unassigned_new
|
||||||
|
|
||||||
|
for key in sorted(curr_dict.keys()):
|
||||||
|
stream = curr_dict[key]
|
||||||
|
if len(stream_assign[stream]) > 1:
|
||||||
|
if not check_sharing(stream_assign[stream]):
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
if debug:
|
if debug:
|
||||||
print(stream_assign)
|
print(stream_assign)
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user