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:
Andrew Tridgell 2021-03-02 08:52:11 +11:00
parent 6cb263056d
commit 4fbdb59398

View File

@ -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)