2022-05-15 21:39:29 +00:00
""" Used by the make system to generate keyboard.c from info.json.
"""
from milc import cli
from qmk . info import info_json
from qmk . commands import dump_lines
from qmk . keyboard import keyboard_completer , keyboard_folder
from qmk . path import normpath
2024-06-18 02:44:22 +00:00
from qmk . constants import GPL2_HEADER_C_LIKE , GENERATED_HEADER_C_LIKE , JOYSTICK_AXES
2022-05-15 21:39:29 +00:00
2024-01-14 10:33:58 +00:00
def _gen_led_configs ( info_data ) :
lines = [ ]
if ' layout ' in info_data . get ( ' rgb_matrix ' , { } ) :
lines . extend ( _gen_led_config ( info_data , ' rgb_matrix ' ) )
if ' layout ' in info_data . get ( ' led_matrix ' , { } ) :
lines . extend ( _gen_led_config ( info_data , ' led_matrix ' ) )
return lines
def _gen_led_config ( info_data , config_type ) :
2022-05-15 21:39:29 +00:00
""" Convert info.json content to g_led_config
"""
cols = info_data [ ' matrix_size ' ] [ ' cols ' ]
rows = info_data [ ' matrix_size ' ] [ ' rows ' ]
lines = [ ]
2023-02-10 22:39:13 +00:00
matrix = [ [ ' NO_LED ' ] * cols for _ in range ( rows ) ]
2022-05-15 21:39:29 +00:00
pos = [ ]
flags = [ ]
2023-02-10 22:39:13 +00:00
led_layout = info_data [ config_type ] [ ' layout ' ]
for index , led_data in enumerate ( led_layout ) :
if ' matrix ' in led_data :
row , col = led_data [ ' matrix ' ]
matrix [ row ] [ col ] = str ( index )
pos . append ( f ' {{ { led_data . get ( " x " , 0 ) } , { led_data . get ( " y " , 0 ) } }} ' )
flags . append ( str ( led_data . get ( ' flags ' , 0 ) ) )
2022-05-15 21:39:29 +00:00
if config_type == ' rgb_matrix ' :
lines . append ( ' #ifdef RGB_MATRIX_ENABLE ' )
lines . append ( ' #include " rgb_matrix.h " ' )
elif config_type == ' led_matrix ' :
lines . append ( ' #ifdef LED_MATRIX_ENABLE ' )
lines . append ( ' #include " led_matrix.h " ' )
lines . append ( ' __attribute__ ((weak)) led_config_t g_led_config = { ' )
lines . append ( ' { ' )
for line in matrix :
2023-02-10 22:39:13 +00:00
lines . append ( f ' {{ { " , " . join ( line ) } }} , ' )
2022-05-15 21:39:29 +00:00
lines . append ( ' }, ' )
2023-02-10 22:39:13 +00:00
lines . append ( f ' {{ { " , " . join ( pos ) } }} , ' )
lines . append ( f ' {{ { " , " . join ( flags ) } }} , ' )
2022-05-15 21:39:29 +00:00
lines . append ( ' }; ' )
lines . append ( ' #endif ' )
2024-01-14 10:33:58 +00:00
lines . append ( ' ' )
2022-05-15 21:39:29 +00:00
return lines
2023-10-29 00:09:02 +00:00
def _gen_matrix_mask ( info_data ) :
""" Convert info.json content to matrix_mask
"""
cols = info_data [ ' matrix_size ' ] [ ' cols ' ]
rows = info_data [ ' matrix_size ' ] [ ' rows ' ]
2023-11-01 00:55:48 +00:00
# Default mask to everything disabled
2023-12-05 13:32:07 +00:00
mask = [ [ ' 0 ' ] * cols for _ in range ( rows ) ]
2023-10-29 00:09:02 +00:00
2023-11-01 00:55:48 +00:00
# Mirror layout macros squashed on top of each other
2023-12-05 13:32:07 +00:00
for layout_name , layout_data in info_data [ ' layouts ' ] . items ( ) :
2023-11-01 00:55:48 +00:00
for key_data in layout_data [ ' layout ' ] :
row , col = key_data [ ' matrix ' ]
2023-12-05 13:32:07 +00:00
if row > = rows or col > = cols :
cli . log . error ( f ' Skipping matrix_mask due to { layout_name } containing invalid matrix values ' )
return [ ]
2023-11-01 00:55:48 +00:00
mask [ row ] [ col ] = ' 1 '
2023-10-29 00:09:02 +00:00
lines = [ ]
lines . append ( ' #ifdef MATRIX_MASKED ' )
lines . append ( ' __attribute__((weak)) const matrix_row_t matrix_mask[] = { ' )
for i in range ( rows ) :
lines . append ( f ' 0b { " " . join ( reversed ( mask [ i ] ) ) } , ' )
lines . append ( ' }; ' )
lines . append ( ' #endif ' )
return lines
2024-06-18 02:44:22 +00:00
def _gen_joystick_axes ( info_data ) :
""" Convert info.json content to joystick_axes
"""
if ' axes ' not in info_data . get ( ' joystick ' , { } ) :
return [ ]
axes = info_data [ ' joystick ' ] [ ' axes ' ]
axes_keys = list ( axes . keys ( ) )
lines = [ ]
lines . append ( ' #ifdef JOYSTICK_ENABLE ' )
lines . append ( ' joystick_config_t joystick_axes[JOYSTICK_AXIS_COUNT] = { ' )
# loop over all available axes - injecting virtual axis for those not specified
for index , cur in enumerate ( JOYSTICK_AXES ) :
# bail out if we have generated all requested axis
if len ( axes_keys ) == 0 :
break
axis = ' virtual '
if cur in axes :
axis = axes [ cur ]
axes_keys . remove ( cur )
if axis == ' virtual ' :
lines . append ( f " [ { index } ] = JOYSTICK_AXIS_VIRTUAL, " )
else :
lines . append ( f " [ { index } ] = JOYSTICK_AXIS_IN( { axis [ ' input_pin ' ] } , { axis [ ' low ' ] } , { axis [ ' rest ' ] } , { axis [ ' high ' ] } ), " )
lines . append ( ' }; ' )
lines . append ( ' #endif ' )
return lines
2022-05-15 21:39:29 +00:00
@cli.argument ( ' -o ' , ' --output ' , arg_only = True , type = normpath , help = ' File to write to ' )
@cli.argument ( ' -q ' , ' --quiet ' , arg_only = True , action = ' store_true ' , help = " Quiet mode, only output error messages " )
@cli.argument ( ' -kb ' , ' --keyboard ' , arg_only = True , type = keyboard_folder , completer = keyboard_completer , required = True , help = ' Keyboard to generate keyboard.c for. ' )
@cli.subcommand ( ' Used by the make system to generate keyboard.c from info.json ' , hidden = True )
def generate_keyboard_c ( cli ) :
""" Generates the keyboard.h file.
"""
kb_info_json = info_json ( cli . args . keyboard )
# Build the layouts.h file.
keyboard_h_lines = [ GPL2_HEADER_C_LIKE , GENERATED_HEADER_C_LIKE , ' #include QMK_KEYBOARD_H ' , ' ' ]
2024-01-14 10:33:58 +00:00
keyboard_h_lines . extend ( _gen_led_configs ( kb_info_json ) )
2023-10-29 00:09:02 +00:00
keyboard_h_lines . extend ( _gen_matrix_mask ( kb_info_json ) )
2024-06-18 02:44:22 +00:00
keyboard_h_lines . extend ( _gen_joystick_axes ( kb_info_json ) )
2022-05-15 21:39:29 +00:00
# Show the results
dump_lines ( cli . args . output , keyboard_h_lines , cli . args . quiet )